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内 容 简 介 


本 书 从 实战 出 发 ， 全 面 、 系 统 地 介绍 了 微软 新 发 布 的 ASPNET 4.0 网 络 开发 基础 、 相 关 开发 框架 及 
应 用 。 书 中 提供 了 大 量 实例 ， 并 提供 了 1 个 通用 模块 和 3 个 源 自 一 线 的 项 目 开 发 案例 供 读者 实战 演练 。 
本 书 附带 1 张 光盘 ， 内 容 为 本 书 涉及 的 源 代码 和 配套 的 教学 视频 ， 另 外 还 赠送 了 C#、ASP.NET 入 门 教学 


视频 等 其 他 学 习 资 料 。 


本 书 共 分 3 篇 。 第 1 篇 介绍 了 模板 页 、 主 题 、Web 服务 、 用 户 控 件 、 自 定义 控件 、ADO.NET 数据 库 
访问 技术 , ASPNET 数据 控件 、 源码 管理 、 三 层 结构 、 单 元 测试 及 搜索 引擎 优化 等 Web 开发 的 关键 技术 ; 
第 2 篇 介绍 了 Visual Studio 2010 新 特性 、LINQ 与 实体 框架 Entity Framework、ASP.NET AJAX 框架 、 优 
秀 的 JavaScrip 框架 jQuery 等 内 容 ; 第 3 篇 介绍 了 1 个 通用 权限 管理 系统 的 开发 ， 另 外 ， 重 点 介绍 了 县 长 
公开 电话 受理 系统 、 社 保 卡 结算 系统 和 新 农 合 管理 系统 3 个 实际 项 目的 开发 过 程 ， 这 3 个 项 目 都 是 作者 


开发 的 拥有 知识 产权 的 项 目 ， 对 提高 读者 的 项 目 开 发 实战 水 平 有 很 大 帮助 。 


本 书 内 容 丰 富 , 重点 突出 , 适合 有 C# 语 言 基础 的 ASPNET 网 络 开发 人 员 阅 读 , 尤其 适合 想 提高 实际 


项 目 开 发 水 平 的 人 员 阅 读 。 另 外 ， 本 书 实用 性 强 ， 很 适合 相关 培训 学 校 的 学 员 作为 教材 使 用 。 


本 书 封面 贴 有 清华 大 学 出 版 社 防伪 标签 ， 无 标签 者 不 得 销售 。 
版 权 所 有 ， 侵 权 必 究 。 侵 权 举报 电话 : 010-62782989 13701121933 


图 书 在 版 编目 (CIP) 数据 


精通 ASP.NET 4.0 网 络 编程 : 基础 、 框 架 与 项 目 实战 / 孙 继 各 等 编著 . 一 北京 : 


出 版 社 ，2011.1 
ISBN 978-7-302-24122-5 


1 . @ 精 … I. @ 孙 … ”II @ 主 页 制作 -程序 设计 JIV. CDTP393.092 
中 国 版 本 图 书馆 CIP 数据 核 字 (2010) 第 232254 号 


责任 编辑 ， 夏 兆 谨 

责任 校对 : 徐 俊 伟 

责任 印 制 : 

出 版 发 行 : 清华 大 学 出 版 社 地  ” 址 : 北京 清华 大 学 学 研 大 厦 A 座 
http://www.tup.com.cn 邮 ” 编 : 100084 
社 总 机 : 010-62770175 邮 ” 购 : 010-62786544 
投稿 与 读者 服务 : 010-62795954, jsjjc@tup.tsinghua.edu.cn 
质量 反馈 : 010-62772015，zhiliang@tup.tsinghua.edu.cn 


经 ” 销 : 全 国 新 华 书店 
开 ”本 : 185X260 印 ” 张 : 36.5 字 ” 数 : 908 千 字 


〈 附 光盘 1 张 ? 
版 ”次 : 2011 年 1 月 第 1 版 印 ”次 : 2011 年 1 月 第 1 次 印刷 
印 ” 数 : 1 一 000 
定 价 : 元 


清华 大 学 


了 中 


前 


为 什么 要 写 这 本 书 ? 


目前 市 场 上 ASP.NET 类 的 图 书 很 多 ， 也 有 少数 可 以 称 得 上 经 典 和 精品 的 图 书 ， 但 是 
能 够 将 技术 与 实际 项 目 开 发 很 好 地 结合 起 来 讲解 的 书 却 是 凤毛麟角 。 这 主要 是 由 于 真实 项 
目 大 都 涉及 软件 公司 或 者 用 户 的 知识 产权 和 商业 机 密 ， 不 能 公开 出 版 。 因 此 ， 我 们 在 市 场 
上 见 到 的 各 种 ASP.NET 开发 类 图 书 中 的 例子 ， 尤 其 是 最 后 的 综合 案例 ， 都 是 类 似 于 教学 
案例 性 质 的 演示 程序 而 非 实际 应 用 案例 。 例 如 ， 我 们 经 常 看 到 的 论坛 系统 、 网 上 书店 、 网 
上 购物 、 医 院 管理 系统 等 。 这 种 案例 规模 小 ， 功 能 不 全 面 ， 界 面 不 细腻 ， 软 件 可 靠 性 不 高 。 
从 各 个 技术 角度 来 说 ， 这 种 以 教学 和 演示 为 目的 的 案例 与 真实 项 目 开 发 都 有 很 大 差距 。 

当然 ， 各 种 类 型 的 ASP.NET 图 书 都 有 自己 的 长 处 和 适合 的 读者 定位 ， 如 前 面 所 说 的 
简化 的 以 教学 为 目的 的 案例 ， 其 优势 在 于 门槛 低 ， 涉 及 技术 少 ， 程 序 结构 简单 ， 容 易 理 解 ， 
适合 于 当 学 生 教材 或 者 给 没有 软件 开发 基础 的 读者 阅读 。 而 对 于 已 经 掌握 了 基本 的 C# 语 言 
和 ASP.NET 基本 语法 的 读者 来 说 ， 他 们 更 希望 有 一 本 能 将 具体 技术 和 实际 项 目 开发 很 好 
地 结合 起 来 ， 指 导 他 们 提高 项 目 实战 开发 水 平 的 书 ， 这 就 需要 以 真实 项 目 案例 为 背景 指导 
读者 学 习 。 

为 了 帮助 缺乏 项 目 经 验 的 读者 深入 理解 真实 的 软件 项 目 开 发 ， 笔 者 挑选 了 几 个 曾经 做 
过 的 实际 项 目 ， 从 需求 、 设 计 、 实 现 、 测 试 几 个 过 程 进行 讲解 ， 帮 助 读 者 理解 项 目 开 发 。 
为 了 让 读者 能 比较 好 地 理解 项 目 开发 ， 本 书 前 半 部 分 先 重点 介绍 了 相关 项 目 中 用 到 的 
ASP.NET 开发 技术 和 相关 框架 , 最 后 提供 了 1 个 通用 模块 和 3 个 源 自 一 线 的 项 目 开 发 案例 
供 读者 实战 演练 。 这 3 个 案例 都 是 作者 近 两 年 设计 开发 的 拥有 知识 产权 的 真实 项 目 ， 对 提 
高 读者 的 项 目 开发 实战 水 平 有 很 大 帮助 。 

本 书 的 写作 和 出 版 受到 滨州 学 院 科研 基金 的 赞助 ， 项 目 编号 为 BZXYG0905。 


本 书 有 何 特色 ? 


1. 紧 跟 行 业 发 展 ， 关 注 最 新 技术 


本 书 对 Visual Studio 2010 /ASPNET 4.0/ C# 4.0 /ADONET 4.0 中 出 现 的 新 技术 进行 了 
讲解 ， 如 集成 开发 环境 新 功能 、C# 4.0 新 特性 、LINQ、Entity Framework、AJAX 等 。 


2. 技术 全 面 ， 讲 解 深入 、 透 彻 
本 书 比较 全 面 、 系 统 地 介绍 了 ASP.NET 网 络 编程 所 涉及 的 关键 技术 ， 并 对 所 涉及 的 


前 言 


第 三 方 框架 做 了 重点 深入 、 透 彻 的 讲解 。 这 些 内 容 并 不 只 局 限于 ASP.NET 本 身 ， 还 涉及 
其 他 客户 端 脚本 技术 、 软 件 设计 思想 、 软 件 开发 规范 等 知识 。 


3. 内 容 有 所 取舍 ， 做 到 重点 突出 


本 书 不 讲解 C# 语 言 基础 和 太 多 的 ASPNET 语法 基础 ， 而 是 关注 于 控件 和 页 面 的 高 级 
应 用 及 实现 原理 ， 尤 其 对 各 种 开发 技术 在 实际 项 目 开 发 中 的 应 用 做 了 重点 介绍 。 


4. 精 选 真实 项 目 案例 ， 提 供 完整 的 源 代 码 ， 超 级 实用 


本 书 精 选 了 1 个 阶段 性 的 项 目 案例 〈 网 上 书店 ) 、1 个 通用 模块 系统 〈 通 用 权限 管理 
系统 ) 和 3 个 拥有 自主 知识 产权 的 真实 项 目 案例 〈 县 长 公开 电话 受理 系统 、 社 保 卡 结算 系 
统 、 新 农 合 管理 系统 ) 进行 讲解 ， 并 提供 了 完整 的 源 代 码 ， 内 容 非常 实用 。 通 过 这 些 案例 ， 
读者 可 以 深入 理解 项 目 开 发 的 过 程 ， 提 升 项 目 开 发 水 平 。 

5. 配 超 值 DVD 光盘 


本 书 配 带 1 张 非常 超 值 的 DVD 光盘 ， 内 容 如 下 : 

本 书 配套 多 媒体 教学 视频 ; 

本 书 所 涉及 的 源 代码 ; 

C# 入 门 教学 视频 (免费 赠送 )， 

ASP.NET 入 门 教学 视频 〈 免 费 赠送 ) ; 

其 他 学 习 资料 〈 免 费 赠送 ) 。 

各 说明 : 配 书 光盘 中 提供 了 县 长 公开 电话 受理 系统 、 社 保 卡 结算 系统 两 个 项 目的 全 部 源 
码 。 新 农 合 项 目 只 提供 了 部 分 代码 。 因 为 该 系统 包含 两 个 相对 独立 的 子 系统 ， 其 
中 一 个 子 系统 是 使 用 WinForm 开发 的 ， 该 部 分 内 容 书 中 没有 介绍 ， 所 以 没有 提 
供 该 部 分 的 源 代码 。 
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本 书 内 容 及 知识 体系 


第 1 篇 ”ASP.NET 网 络 开发 关键 技术 〈 第 1~6 章 ) 


本 篇 介绍 了 ASPNET 网 络 编程 必须 要 掌握 的 一 些 关 键 技术 。 主 要 内 容 包 括 ASP.NET 
网 络 开发 基础 知识 ， 如 ASPNET 事件 模型 、 页 面 生命 周期 、 模 板 页 、 主 题 、Web 服务 、 
用 户 控件 、 自 定义 控件 等 ADO.NET 数据 库 访问 技术 ， 如 连接 数据 库 、 修 改 数据 、 查 询 
数据 、 储 存 过 程 等 , ASPNET 数据 控件 , 如 GridView 控件 、DataList 控件 、 数 据 源 控件 等 ; 
1 个 阶段 性 项 目 案例 网 上 书店 的 实现 ;规范 的 软件 开发 ， 介 绍 管理 源码 、 三 层 结构 和 单元 
测试 等 ; 搜索 引擎 优化 ,介绍 URL 重 写 优化 、 正 则 表达 式 与 URL 重 写 、 页 面 内 容 优化 等 。 


第 2 篇 。 开 发 工具 与 第 三 方 框架 (第 7~10 章 ) 


本 篇 介绍 了 ASP.NET 网 络 开发 所 涉及 的 开发 工具 和 第 三 方 框架 的 使 用 。 主 要 内 容 包 
括 Visual Studio 2010/C# 4.0/ASP.NET 4.0 的 新 功能 和 新 特性 ， 如 集成 开发 环境 的 改进 、C# 


I. 


前 言 


对 动态 数据 类 型 的 支持 、ASPNET 4.0 中 的 配置 文件 转换 等 ，LINQ 和 实体 框架 Entity 
Framework 的 使 用 ， 这 是 微软 公司 推出 的 最 新 的 数据 访问 框架 ， 提 出 了 集成 于 语言 中 的 与 
具体 数据 源 相 分 离 的 数据 访问 和 查询 技术 ， 大 大 提高 了 开发 人 员 的 效率 ; AJAX 框架 原理 、 
ASPNET 自 带 的 ASPNET AJAX 核心 组 件 的 使 用 、 微 软 公司 提供 的 AJAX Control Toolkit 
中 的 几 种 典型 控件 的 使 用 ， 通 过 JavaScript 框架 jQuery 实现 丰富 的 动态 页 面 效 果 ， 以 及 用 
jQuery+ASPNET Web Service 构建 AJAX 应 用 等 。 


第 3 篇 ”项目 实战 〈 第 11~14 章 ) 


本 篇 综合 利用 前 面 所 介绍 的 技术 和 思想 ， 讲 解 了 4 个 真实 项 目 案例 的 设计 与 实现 。 主 
要 内 容 包括 通用 权限 管理 系统 ， 可 以 不 经 修改 即 可 应 用 于 各 个 ASP.NET 项 目 ， 实 现 基本 
的 基于 角色 的 权限 管理 ， 县 长 公开 电话 受理 系统 ， 能 够 对 县 长 公开 电话 工作 进行 全 面 的 业 
务 处 理 、 数 据 查 询 、 统 计 、 报 表 等 ， 社 保 卡 结算 系统 ， 用 于 实现 各 个 定点 医疗 机 构 的 社保 
卡 结算 和 对 账 功能 ， 也 包括 各 种 数据 查询 、 统 计 和 报表 等 ， 新 农 合 管理 系统 ， 能 够 对 新 型 
农村 合作 医疗 业务 进行 日 常 管理 ， 如 农民 档案 管理 、 参 合 退 合 管理 、 缴 费 管理 、 报 销 结 
算 等 。 


适合 阅读 本 书 的 读者 


本 书 假定 读者 已 经 具备 了 一 定 的 编程 基础 ， 掌 握 了 C# 语 言 、SQL 语句 和 SQL Server 
数据 库 的 使 用 ， 所 以 书 中 没有 涉及 太 多 的 基本 语法 、 基 本 控件 和 基本 开发 环境 操作 等 内 容 
的 讲解 ， 而 是 把 重点 放 在 了 关键 技术 、 框 架 和 项 目 实战 上 。 如 果 您 还 不 具备 相关 的 基础 ， 
请 首先 阅读 相关 书籍 ， 打 好 基础 ， 才 能 比较 流畅 地 阅读 本 书 。 本 书 适合 的 读者 如 下 : 
具备 基本 的 ASP.NET 知识 ， 想 进一步 学 习 和 提高 的 人 员 ; 
开发 过 C/S 结构 程序 ， 想 学 习 B/S 开发 的 人 员 ; 

使 用 ASP.NET 开发 Web 应 用 的 程序 员 ; 
想 提高 Web 项 目 开 发 水 平 的 程序 员 ; 
大 中 专 院 校 和 培训 班 的 学 生 。 


本 书 作 者 及 编 委 会 成 员 
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本 书 由 孙 继 可 主 笔 编写 。 其 他 参与 编写 的 人 员 有 班 志 杰 、 陈 旭 、 陈 永 俊 、 陈 争光 、 戴 
建华 、 方 文 票 、 冯 玉 荣 、 高 姗 姗 、 巩 宁 来 、 谷 世 江 、 胡 其 吐 、 黄 飞龙 、 蒋 晓 捷 、 李 德 明 、 
李 显 亮 、 李 志 勇 、 刘 雁 征 、 吕 小 波 、 马 东 、 重 庆 海 、 唐 勇 、 王 浩 、 王 玲 玉 、 王 志 娟 、 武 娜 、 
徐 晓 娟 、 闫 树 丰 、 杨 朝 宇 、 翟 闯 等 。 在 此 表示 感谢 ! 

本 书 编 委 会 成 员 有 欧 振 旭 、 陈 杰 、 陈 冠军 、 项 宇 峰 、 张 帆 、 陈 刚 、 程 彩 红 、 毛 红 娟 、 
聂 庆 亮 、 王 志 娟 、 武 文 娟 、 颜 盟 盟 、 姚 志 娟 、 尹 继 平 、 张 昆 、 张 薛 。 
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第 1 章 ASPNET 网 络 开发 基础 


本 章 将 介绍 ASPNET 的 几 个 关键 技术 ， 包 括 服务 器 端 控件 事件 模型 、 页 面 生命 周期 、 
母 版 、 用 户 控件 和 自 定义 控件 等 。 阅 读本 书 的 读者 应 该 对 ASPNET 有 基本 的 了 解 ， 熟 悉 
Visual Studio 开发 环境 ， 能 够 使 用 Visual Studio (或 者 Visual Web Developer Express) 创建 
ASPNET Web 应 用 程序 并 且 运 行 ， 了 解 ASPNET 常用 的 基本 控件 ， 如 Button、TextBox、 
DropDownList 等 。 


1.1 ASPNET 事件 模型 和 页 面 生命 周期 


ASPNET 对 Web 页 面 控件 的 事件 进行 了 封装 , 从 而 允许 开发 人 员 用 一 种 类 似 于 处 理 本 
地 WinForm 窗 体 事件 的 方式 来 处 理 Web 页 面 事件 ， 大 大 简化 了 开发 流程 。 任 何事 物 都 有 
两 面 性 ，ASPNET 这 种 封装 也 不 例外 ， 在 带 来 极 大 方便 的 同时 ， 也 使 得 ASPNET 页 面 从 
创建 到 销毁 的 过 程 更 加 复杂 。 

本 节 将 对 ASPNET 服务 器 端 控件 事件 模型 和 ASPNET 页 面 生命 周期 进行 介绍 ， 掌 握 
这 两 个 知识 点 对 于 深入 理解 ASPNET 运行 原理 、 进 行 ASPNET 高 级 开发 有 重要 作用 。 


1.1.1 经 典 的 Web 事件 处 理 方法 


为 了 理解 ASPNET 服务 器 控件 事件 模型 及 其 优点 ， 先 要 理解 Web 应 用 程序 的 请 求 
响应 模型 及 事件 处 理 方式 。 

Web 应 用 程序 或 者 网 站 包含 两 个 重要 角色 : 服务 器 和 浏览 器 (客户 端 )。 用 户 通 过 浏 
览 器 向 Web 服务 器 发 送 HTTP 请 求 ， 服 务 器 响应 此 请 求 并 且 向 浏览 器 返回 HTML 代码 ， 
HTML 代码 到 达 浏 览 器 后 在 浏览 器 中 以 可 视 化 的 形式 显示 出 来 ， 这 个 过 程 如 图 1.1 所 示 。 


ml HTTP 请 求 和 


HTTP 响 应 (如 HTME) 
浏览 器 


Web 服 务 器 


图 1.1 Web 请 求 


响应 模型 


用 户 在 浏览 器 中 进行 的 操作 ， 如 单 击 一 个 按钮 等 ， 也 会 转化 为 一 个 HTTP 请 求 〈 通 常 
是 一 个 POST) 发 送 到 服务 器 ， 然 后 Web 服务 器 处 理 此 请 求 并 且 返 回响 应 内 容 。 这 个 过 程 
与 前 面 所 述 的 请 求 一 一 响应 过 程 完全 相同 。 
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对 于 ASPNET 开发 人 员 来 说 ， 所 开发 的 ASPNET 程序 部 署 在 Web 服务 器 上 ， 通 过 某 
种 机 制 〈 通 常 是 IS) 响应 和 处 理 客户 端 请 求 。 从 上 面 的 讨论 可 以 看 出 ， 在 Web 应 用 程序 
中 ， 开 发 人 员 需 要 针对 各 种 不 同 的 HTTP 请 求 来 编写 代码 ， 而 与 浏览 器 端的 事件 〈 如 按钮 
的 单 击 ) 无 关 。 这 种 编程 方式 很 不 直观 ， 客 户 端 〈 浏 览 器 ) 事件 与 服务 器 端 事件 的 处 理 是 
完全 分 开 的 ， 下 面 通过 一 个 例子 具体 说 明 此 问题 。 

【 例 1-1】 经 典 的 Web 事件 处 理 。 

本 例 演示 在 传统 Web 编程 中 如 何 处 理 客户 端 事件 。 在 本 例 中 , 用 户 在 浏览 器 端 单 击 一 
个 按钮 ， 则 服务 器 响应 此 按钮 的 单 击 事件 并 返回 服务 器 当前 时 间 。 

完成 这 个 例子 的 具体 操作 过 程 如 下 。 

(1) 在 Visual Studio 中 创建 一 个 ASPNET Web 应 用 程序 , 项 目 命名 为 ClassicWebApp。 
项 目 创建 后 ， 会 自动 生成 一 个 Default.aspx 的 ASPNET Web 窗 体 ， 由 于 本 例 采 用 基于 纯粹 
的 HTTP 请 求 的 方式 处 理 页 面 事件 ， 而 不 使 用 ASPNET Web 窗 体 ， 所 以 把 Default.aspx 从 
项 目 中 删除 。 


且 提 示 : 本 书 假定 读者 已 经 熟悉 了 Visual Studio 开发 环境 和 基本 操作 ， 如 创建 项 目 、 运 行 
项 目 等 ,为 了 突出 重点 和 节省 篇 幅 ， 书 中 对 于 此 类 基本 操作 不 给 出 具体 操作 过 程 
和 界面 截图 。 如 果 读 者 对 此 不 熟悉 ， 可 参考 其 他 ASP.NET 编程 资料 。 


(2) 在 项 目 中 添加 一 个 “一 般 处 理 程序 ”ASPNET 的 一 般 处 理 程序 是 一 个 实现 了 
IHttpHandler 接口 的 类 ， 可 以 处 理 HTTP 请 求 。 

添加 一 般 处 理 程序 的 具体 操作 过 程 是 ， 选择 菜单 中 的 “项 目 ”|“ 添 加 新 项 ”， 在 弹出 
的 “添加 新 项 ”对 话 框 中 选择 “一 般 处 理 程 序 ” 并 重 命 名 为 MyHandler， 如 图 1.2 所 示 。 


区 上 7 rw 
区 RT in te Bs 


图 1.2 添加 一 般 处 理 程序 


(3) 添 加 了 一 般 处 理 程序 MyHandler 类 后 , 可 以 看 到 其 中 有 一 个 ProcessRequest() 方 法 ， 
代码 如 下 : 
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public void ProcessRequest (HttpContext context) 
. 


context .Response.ContentType = "text/plain™"; 

context .Response.Write ("Hello World"); 

这 个 方法 就 是 用 于 处 理 浏 览 器 发 送 过 来 的 HITP 请 求 的 方法 。 方法 有 一 个 HttpContext 
类 型 的 参数 , 代表 了 当前 HTTP 请 求 上 下 文 , 可 以 通过 这 个 属性 向 客户 端 返 回 特 定 的 内 容 。 
本 例 向 客户 端 返回 服务 器 的 当前 日 期 和 时 间 ， 如 下 代码 所 示 。 

a void ProcessRequest (HttpContext context) 


// 设 置 HTTP 响应 类 型 为 HTML 文本 

context.Response.ContentType = "text/html"; 

/ /构建 一 个 HTML 格式 的 包含 当前 服务 器 时 间 的 字符 串 

string s = string.Format ("<html> <body> <h3> 服务 器 当前 时 间 为 : <span style 
= \"color:green;\" > {0:G} <span> </h3> </body> </html>", DateTime.Now); 
context .Response.Write (s); // 向 浏览 器 返回 服务 器 这 个 字符 串 


(4) 在 项 目 中 添加 一 个 HIML 页 面 ， 命 名 为 MyPage.htm。 
(5) 在 MyPage.htm 中 , 添加 一 个 表单 Form， ae 光村 属性 为 MyHandlerashx， 


从 而 使 表单 提交 给 MyHandlerashx， 在 表单 中 放 一 个 提交 按钮 ， 以 执行 提交 并 获取 服务 器 
当前 时 间 。MyPage.htm 页 面 的 关键 代码 如 下 : 
<body> 


<form action="MyHandler.ashx"> 

<input type="submit" value=" 获 取 服 务 器 时 间 "” /> 

</form> 

</body> 

(6) 右 击 MyPage.htm 页 面 空白 处 ， 从 弹出 的 快捷 菜单 中 选择 “在 浏览 器 中 查看 ” 选 
项 ， 则 可 以 运行 此 页 面 ， 运 行 结果 如 图 1.3 所 示 。 

(7) 在 MyPagehtm 页 面 上 单 击 “获取 服务 器 时 间 ” 按 钮 ， 则 浏览 器 会 跳 转 到 
MyHandler.ashx 页 面 〈 见 提示 )， 显 示 结 果 如 图 1.4 所 示 。 


文件 四 CE EECTTOETOEEEEE FE 

> Cn -. lea ew/ .leap 
Google | ssp net vebtom -| -1»% -| Goosk -1:1.»%-.@- 
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服务 器 当前 时 间 为 : 2010-01-19 12:33:05 


EslES WED 更 
图 1.3 MyPage.htm 运行 界面 图 1.4 MyHandler.ashx 页 面 


全 提示 : 严格 来 说 ，MyHander ashx 并 不 是 一 个 普通 意义 上 的 页 面 ， 但 是 在 浏览 器 看 来 ， 
这 是 一 个 合法 的 URL， 可 以 访问 并 且 能 够 得 到 用 于 显示 的 HTML 代码 ， 与 其 他 
页 面 没有 本 质 区 别 。 


在 例 1-1 中 ，MyHanlder.ashx 处 理 HTTP 请 求 时 ， 总 是 返回 服务 器 当前 日 期 和 时 间 ， 


4. 
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而 没有 考虑 MyPage.htm 页 面 上 的 内 容 。 在 实际 应 用 中 ， 服 务 器 端 通常 需要 根据 页 面 上 的 
内 容 做 出 相应 的 处 理 。 例 如 ， 对 于 用 户 登 录 程 序 来 说 ， 服 务 器 后 台 处 理 程序 需要 获取 用 户 
在 登录 页 面 中 输入 的 用 户 名 和 密码 ， 进 而 判断 是 否 合法 用 户 。 还 有 一 些 页 面 ， 其 中 包含 多 
个 按钮 , 而 每 个 按钮 的 作用 是 不 同 的 , 服务 器 端 需要 区 分 用 户 在 浏览 器 中 单 击 了 哪个 按钮 ， 
从 而 进行 不 同 的 处 理 。 

对 于 ASPNET Web 应 用 程序 来 说 , 服务 器 端 代码 可 以 通过 HttpContext.Request.Params 
属性 获得 用 户 在 浏览 器 页 面 中 输入 的 内 容 ， 下 面 通过 一 个 例子 来 演示 具体 实现 。 

【 例 1-2】 处 理 浏览 器 端 输 入 事件 。 

本 例 演示 在 ASPNET 中 如 何在 服务 器 端 获得 浏览 器 页 面 上 输入 的 内 容 , 并 根据 用 户 单 
击 不 同 的 按钮 而 执行 不 同 操作 。 

(1) 创建 一 个 ASPNET Web 应 用 程序 ， 删 除 默认 的 Default.aspx 页 面 。 

(2) 在 项 目 中 添加 一 个 用 户 登 录 HTML 页 面 ， 命 名 为 login.htm， 关 键 代 码 如 下 : 

<body> 

<form action="LoginHandler.ashx"> 

用 户 名 : <input type="text" name="username" /><br /> 

密码 : <input type="password" name="password" /><br /> 

<!-- 注 意 : 下 面 这 两 个 按钮 的 name 要 相同 ， 以 便于 在 服务 器 端 获取 单 击 了 哪个 按钮 --> 

<input type="submit" name="button1" value=" 登 录 " /> 

<input type="submit" name="buttonl" value=" 注 册 " /> 


</form> 
</body> 


(3) 在 项 目 中 添加 一 般 处 理 程序 ， 命 名 为 LoginHandler， 为 其 ProcessRequest() 方 法 编 
写 代码 ， 根 据 不 同情 况 处 理 Web 请 求 。 


public void ProcessRequest (HttpContext context) 
{ // 获 取 页面 中 输入 的 用 户 名 ， 其 中 Params 后 面 的 键 为 页 面 上 控件 的 名 称 


string user = context.Request.Params["username"]; 
string pass = context.Request.Params["password"]; // 获 取 页 而 中 输入 的 密码 
string button = context.Request.Params["button1"]; // 获 取 所 单 击 的 按钮 


context .Response.ContentType = "text/plain"; // 设 置 响应 类 型 为 纯 文本 
// 如 果 单 击 “ 注 册 ” 按 钮 引起 的 提交 ， 则 提示 功能 未 实现 

if (button == "注册 ") 

{ 


context .Response.Write ("注册 功能 尚未 实现 ..."); 
} 
// 如 果 单 击 “ 登 录 ” 按 钮 引起 提交 ， 则 判断 输入 的 用 户 名 和 密码 是 否 正 确 
else if (button == "登录 ") 
{ 
if (user.ToLower() == "sunjilei" && pass == "123456") 
context .Response.Write ("登录 成 功 ! "); 
else 


context .Response.Write ("用 户 名 或 密码 不 正确 ， 登 录 失败 ! "); 
上 
// 如 果 不 是 单 击 “ 注 册 ” 或 “登录 ”按钮 ， 则 认为 是 不 可 识别 的 命令 


else 
{ 

context .Response.Write ("LoginHandler 不 能 识别 接收 到 的 命令 。"+button); 
} 
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(4) 运行 项 目 ， 运 行 结果 如 图 1.5 所 示 。 
= x| 


文件 四 ”编辑 包 ) 查看 中 ”历史 书签 @) 工具 QD 才 助 名 

Ee 全 让 一 - | 加 :ae 有 户 
asp. net http referer "13-1» SS%-@- 
| 


A | ] 
宙 凤 ,也 66666 | 
| -注册 | 


图 1.5 例 1-2 运行 界面 


在 上 述 代 码 中 ， 有 两 个 地 方 需要 说 明 。 

(1) 可 以 通过 contextResponse.Params["username"] 得 到 用 户 在 浏览 器 页 面 中 输入 的 登 
录用 户 名 ， 其 中 usemame 为 HTML 页 面 上 用 户 名 文本 控件 的 名 称 。 可 以 通过 同样 的 方法 
得 到 页 面 上 其 他 控件 的 值 。 

(2) 由 于 此 页 面 有 两 个 提交 按钮 ， 一 个 “登录 ”， 一 个 “注册 ” 两 者 功能 不 同 。 在 同 
一 个 处 理 程序 LoginHandler 中 ， 必 须 区 分 这 两 个 按钮 。 这 里 使 用 了 一 个 小 技巧 ， 即 把 两 个 
按钮 的 名 称 (name 属性 ) 设置 为 相同 (都 是 button1)， gs (value 属性 ) 不 相 
同 ， 然 后 在 服务 器 端 就 可 以 通过 获取 buttonl 控件 的 值 来 判断 单 击 了 哪个 按钮 。 


1.1.2 ASP.NET 服务 器 控件 事件 模型 


从 1.1.1 节 的 两 个 例子 可 以 看 出 , 在 传 
一 个 页 面 所 有 控件 的 事件 通常 都 由 一 个 程序 来 处 理 ， 事 件 处 理 程序 与 事件 之 间 没 有 直接 联 
系 。 另 一 方面 ， 要 想 获得 页 面 上 输入 的 内 容 也 不 方便 ， 需 要 采用 Request.Params[" 控 件 名 "] 
这 种 形式 。 由 于 控件 名 是 一 个 字符 串 ， 不 能 利用 智能 提示 ， 不 能 进行 编译 检查 ， 所 以 很 容 
易 写 错 。 


响应 的 底层 细节 , 从 而 很 好 地 解决 
也 上 述 问题 ， 使 控件 事件 与 ea ， 使 得 控件 的 访问 更 安全 高 效 。 

ASPNET 服务 器 端 控件 事件 模型 与 传统 的 WinForm 事件 类 似 ， 页 面 上 有 许多 控件 ， 
一 个 控件 有 许多 事件 ， 对 于 每 个 控件 的 每 个 事件 都 可 以 单独 编写 一 个 事件 处 理 程序 ， 当 控 
件 发 生 某 个 事件 时 ， 就 会 执行 对 应 的 事件 处 理 程序 。 

下 面 通 过 一 个 例子 演示 ASPNET 事件 模型 的 优势 。 

【 例 1-3】 登录 页 面 。 

本 例 与 例 1-2 功能 相同 ， 所 不 同 的 是 本 例 采 用 ASPNET Web 窗 体 + 服务 器 端 控件 来 实 
现 。 对 比 同一 功能 的 两 种 不 同 实现 ， 更 能 体现 二 者 差别 。 

(1) 创建 一 个 ASPNET Web 应 用 程序 ， 命 名 为 EventModelSample。 

(2) 在 自动 生成 的 Default.aspx 上 放置 两 个 TextBox 控件 和 两 个 Button 控件 ， 页 面 布 
局 与 例 1-2 相同 。 页 面 代码 如 下 : 


。6。 
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<body> 
<form id="forml" runat="server"> 
<div> 
用 户 名 : <asp:TextBox ID="username" runat="server"></asp:TextBox><br /> 
密码 : <asp:TextBox ID="password" TextMode="Password" runat="server"> 
</asp:TextBox><br /> 
<asp:Button ID="login" runat="server"” Text=" 登 录 " onclick="login 
Ciieck" /> 
<asp:Button ID="register" 
runat="server” Text=" 注 册 " onclick="register Click" /> 


</div> 
</form> 
</body> 
(3) 为 “登录 ”按钮 和 “注册 ”按钮 编写 事件 处 理 程序 ， 代 码 如 下 : 
// 登 录 按 钮 事件 处 理 程序 
protected void login Click(object sender, EventArgs e) 
{ 
// 获 取 用 户 输入 的 用 户 名 和 密码 
String user = username.Text; 
string pass = password.Text; 
if (user.ToLower() == "sunjilei" && pass == "123456") 
Response.Write ("登录 成 功 ! "); 
else 
Response.Write ("用 户 名 或 密码 不 正确 ， 登 录 失 败 ! "); 
// 注 册 按钮 事件 处 理 程序 


protected void register Click(object sender, EventArgs e) 
| 

Response.Write ("注册 功能 尚未 实现 .. ."); 
} 


对 比例 1-3 和 例 1-2 可 以 看 出 ， 在 ASPNET Web 窗 体 中 ， 每 个 控件 的 事件 都 是 单独 的 


方法 ， 各 个 事件 处 理 程序 相互 独立 ， 事 件 与 其 处 理 程序 有 明确 的 对 应 关系 ， 从 而 使 代码 更 
加 清晰 简洁 ， 提 高 了 可 维护 性 。 


i 


3 ASP.NET 页 面 生 命 周期 


一 个 ASPNET 页 面 ， 从 创建 到 销毁 的 过 程 ， 称 为 页 面 的 生命 周期 。 这 是 一 个 短暂 但 却 


复杂 的 过 程 。 当 ASPNET 接收 到 对 于 某 一 页 面 的 请 求 后 ， 此 页 面 就 被 创建 ， 从 而 开始 了 页 
面 的 生命 周期 ， 经 过 一 系列 操作 后 ， 页 面 产生 了 对 应 的 HIML 代码 并 且 返 回 给 浏览 器 ， 页 
面 就 被 销毁 ， 页 面 生命 周期 也 就 结束 。 


一 个 ASPNET 页 面 是 一 个 System.Web.UI.Page 类 的 派生 类 , 在 页 面 从 创建 到 销毁 的 生 


命 周期 中 ，Page 类 会 先后 触发 多 个 事件 。 表 1.1 按照 发 生 的 先后 顺序 列 出 了 Page 类 的 主要 


事件 。 
表 1.1 Page 类 生命 周期 主要 事件 
事件 名 称 说 ”上 明 
sit 页 面 初始 化 以 前 触发， 通常 可 以 在 此 事件 中 检查 IsPostBack 属性 ， 动 态 创建 控件 或 
者 动态 设计 主题 
Init 当 页 面 中 所 有 控件 初始 化 完成 时 触发 ， 可 以 在 此 事件 中 访问 控件 初始 化 属性 
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事件 名 称 说 明 

InitComplete 初始 化 完成 后 触发 ， 到 此 事件 发 生 时 页 面 上 所 有 控件 和 页 面 本 身 的 初始 化 都 已 完成 
二 在 页 面 加 载 前 触发 ， 在 此 事件 中 会 加 载 页 面 和 控件 的 视图 状态 (ViewState)， 并 处 理 
Load 页 面 加 载 事 件 ， 可 以 此 事件 中 设置 控件 属性 

LoadComplete | 控件 及 页 面 加 载 完 成 后 触发 此 事件 

PreRender 在 页 面 即将 呈现 时 触发 ， 可 以 此 事件 中 对 页 面 或 者 控件 外 观 进 行 最 后 的 修改 
Unload 页 面 处 理 完 时 触发 ， 通 常 在 此 事件 中 进行 清理 工作 


在 ASPNET 页 面 生命 周期 中 ， 要 完成 一 些 工作 ,包括 加 载 主 题 、 加 载 视图 状态 、 呈 现 
控件 等 。 从 ASPNET 页 面 要 完成 的 工作 角度 看 ， 页 面 生命 周期 如 图 1.6 所 示 。 


ASPNET 页 面 生命 周期 


[ 门 。 |] 请 求 转交 
给 ASPNET 页 面 和 控件 初始 化 
HTTP 请 求 
ee 1 
加 载 ViewState 
Se 
加 载 回 发 数据 
Web 1 
服务 器 控件 和 页 面 的 Load 事 件 
HTTP 响 应 | (如 TS) EEC 
1 
处 理 回 发 事件 
1 
保存 ViewState 
返回 HTML < 
i 呈现 页 面 
图 1.6 ASP.NET 页 面 生命 周期 的 各 阶段 工作 


下 面 通过 一 个 例子 来 更 加 直观 地 说 明 页 面 及 控件 事件 发 生 的 顺序 。 

【 例 1-4】 页 面 生命 周期 。 

本 例 创 建 一 个 简单 的 页 面 ， 其 中 包含 几 个 控件 ， 当 页 面 的 主要 事件 发 生 时 ， 在 页 面 上 
输出 提示 信息 ， 从 而 演示 页 面 生命 周期 。 

(1) 创建 一 个 ASPNET Web 应 用 程序 ， 命 名 为 PageLifeCycle。 

(2) 在 项 目 中 添加 一 个 页 面 ， 并 参照 以 下 代码 修改 页 面 。 


<body> 


<form id="forml" runat="server"> 


<div> 
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输入 你 的 名 字 : <asp:TextBox ID="yourName" runat="server" 
ontextchanged="yourName TextChanged"></asp:TextBox> 

<asp:Button ID="ok" runat="server" Text=" 确 定 " onclick="ok Click" /> 
<br /> 
<asp:Label ID="hello" runat="server" Text="Label"></asp:Label> 

</div> 

</form> 

</body> 


(3) 为 页 面 上 按钮 的 Click 事件 和 文本 框 的 TextChanged 事件 添加 处 理 程序 ， 代 码 
如 下 : 


protected void ok Click(object sender, EventArgs e) 
. Response.Write ("按钮 Click 事件 <br/>"); 
hello.Text = yourName.Text+"， 你 好 。"; 
ep void yourName TextChanged (object sender, EventArgs e) 
! Response.Write ("文本 杠 Textchanged 事件 <br/>"); 
, 


(4) 重 载 Page 类 与 页 面 生命 周期 相关 的 方法 ， 以 便 在 页 面 事件 发 生 时 输出 提示 信息 ， 
代码 如 下 : 


protected void Page Load(object sender, EventArgs e) //PageLoad 事件 
|! 
Response.Write ("页 面 加 载 事 件 (Page Load) <br/>"); 
1 
protected override void OnLoad(EventArgs e) //onLoad() 方 法 
{ 
Response.Write ("页 面 加 载 事 件 (OnLoad) <br/>"); 
base.OnLoad (e); 
} 
protected override void AddedControl (Control control, int index) 
// 控 件 被 添加 时 调用 
{ 
Response.Write (string.Format ("控件 被 添加 : {0}<br/>",control.GetType(). 
Name)); 
base.AddedControl (control, index); 
protected override void LoadControlState (object savedState) 
// 加 载 控件 状态 
{ 
Response .Write ("加 载 控 件 状 态 (LoadControlState) <br/>"); 
base.LoadControlState (savedState); 
} 
protected override void LoadViewState (object savedState)  // 加 载 视图 状态 
{ 
Response.Write ("加 载 视图 状态 (LoadViewState) <br/>"); 
base.LoadViewState (savedState); 
. 
protected override void OnInit (EventArgs e) // 页 面 初始 化 
| 
Response.Write ("页 面 初始 化 事件 (OnInit) <br/>"); 
base.OnInit (e); 
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protected override void OnLoadComplete (EventRrgs e) /7 页 面 加 载 完成 
{ 
Response.Write ("页 面 加 载 完 成 (OnLoadComplete) <br/>"); 
base.OnLoadComplete (e); 


} 
protected override void OnPreLoad (EventArgs e) // 页 面 加 载 前 
{ 
Response.Write ("页 面 即将 加 载 (OnPreLoad) <br/>"); 
base.OnPreLoad (e); 
} 
protected override void OnPreRender (EventArgs e) // 页 面 呈 现 前 
{ 
Response.Write ("页 面 即将 呈现 (OnPreRender) <br/>"); 
base.OnPreRender (e); 
} 
protected override void OnUnload (EventArgs e) // 页 面 生 载 
{ 


// 在 Unload 事件 中 ，Response.Write 已 经 被 关闭 ， 不 可 使 用 ， 下 面 代码 不 可 执行 
//Response.Write(" 页 面 印 载 (OnUnload) <br/>"); 
base.OnUnload (e); 

1 


(5) 运行 此 项 目 ， 运 行 结果 如 图 1.7 所 示 。 


ET TE 


[0 "| 


rn PE Ey 


oe lie oyele zagle - oooae" | J http://lecalnkr/Defemt ssyz 回 全 


控件 被 添加 ，LiteralControl 


控件 被 添加 ，HtmlHead 

榨 件 蔽 稚 加 ， Cl 
皖 件 被 水 加 ，HtmlFor 

拉 件 被 添加 : LiteralControl 
页 面 初始 化 事件 〔OnInit) 
页 面 即将 加 载 Corprel ead) 
用 面 加 井 事 件 ( 

页 面 加 载 事件 OP 
文本 框 TextChange 

按钮 Click 


输入 你 的 名 宇 ，protm 确定 


John， 你 好 。 
| [rls 


图 1.7 页 面 生命 周期 示例 运行 界面 


1.2 母 版 页 


不 管 是 在 做 网 站 还 是 开发 Web 应 用 程序 ， 为 了 使 作品 在 外 观 上 标准 、 规 范 ,通常 一 个 
项 目 中 的 所 有 页 面 风格 和 结构 是 一 致 的 ， 而 且 各 个 页 面 通常 有 一 部 分 相同 的 地 方 ， 如 公司 
LOGO、 页 面 顶 部 标题 栏 、 页 脚 说 明 信 息 等 。 利 用 ASPNET 的 母 版 技术 ， 可 以 快速 开发 出 
风格 一 致 且 易 于 维护 的 页 面 


1.2.1 母 版 页 的 概念 和 作用 


如 前 所 述 ，Web 应 用 程序 通常 要 求 许多 页 面具 有 外 观 上 的 一 致 性 ， 例 如 具有 相同 的 标 
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题 栏 、LOGO、 页 脚 等 。 为 了 达到 这 种 效果 ， 最 简单 的 做 法 是 把 相同 的 代码 复制 粘贴 到 每 
一 个 页 面 中 。 但 是 这 显然 不 符合 软件 设计 思想 ， 相 同 的 代码 重复 出 现 了 多 次 ， 这 会 给 以 后 
的 修改 造成 很 大 困难 。 

根据 面向 对 象 思想 ， 如 果 许 多 事物 (类 ) 具有 相同 的 功能 ， 那 么 可 以 从 这 些 事件 (类) 
抽象 一 个 基 类 ,把 公共 的 代码 放 到 基 类 中 ，, 再 从 基 类 派生 其 他 类 ,这样 就 实现 了 代码 复 用 。 
而 且 如 果 要 修改 公共 的 功能 ， 只 需要 修改 基 类 一 个 类 就 可 以 。 

把 这 种 抽象 和 继承 的 思想 应 用 于 ASPNET 页 面 的 设计 , 这 就 是 母 版 页 。 母 版 页 就 相当 
于 基 类 ， 其 中 包含 着 公共 的 元 素 ， 从 母 版 页 可 以 创建 其 他 页 面 ， 相 当 于 从 基 类 派生 ， 派 生 
类 自然 就 具有 了 基 类 的 所 有 功能 。 

与 基 类 在 类 层次 结构 中 所 起 的 作用 类 似 ， 母 版 页 在 页 面 设 计 中 就 相当 于 一 个 模板 或 杠 
架 , 在 此 基础 上 ， 可 以 创建 其 他 页 面 。 从 母 版 页 创建 的 页 面 称 为 “内 容 页 ”， 内 容 页 自动 包 
含 了 母 版 页 中 所 有 的 元 素 ， 而 且 还 可 以 添加 自己 的 元 素 以 扩充 母 版 的 内 容 。 

在 类 的 继承 中 ， 基 类 中 包含 虚 方法 ， 派 生 类 可 以 重 写 此 方法 。 与 此 类 似 ， 母 版 页 中 包 
含 称 为 “内 容 占 位 符 ” 的 控件 ContentPlaceHolder， 内 容 页 中 可 以 在 占 位 符 控件 中 按照 自己 
的 要 求 放置 各 种 控件 。 

母 版 页 的 扩展 名 为 .master， 一 个 母 版 页 中 可 以 包含 一 到 多 个 占 位 符 。 当 浏览 器 请 求 一 
个 内 容 页 时 ，ASPNET 会 把 母 版 页 与 内 容 页 合并 ， 再 把 合并 后 的 页 面 发 给 浏览 器 。 这 个 过 
程 如 图 1.8 所 示 。 

母 版 文件 A.master 内 容 页 文件 B.aspx 


<%@ Master ... %> <%@ Page MasterPageFile= "A.master" %> 


<asp:Content ID="head" runat="server" 
ContentPlaceHolderID="head"> 
此 处 为 内 容 </asp:Content> 


<asp:ContentPlaceHolder ID="head" 
runat="server" /> 


<asp:Content ID="content" runat="server" 
ContentPlaceHolderID="head"> 
此 处 为 内 容 </asp:Content> 


ID="content" runat="server" /> 


<asp:ContentPlaceHolder | 


图 1.8 母 版 页 与 内 容 页 


全 注意 : 母 版 页 必须 与 内 容 页 合并 后 作为 一 个 单独 页 面 显示 在 浏览 器 中 。 母 版 页 本 身 不 能 
直接 在 浏览 器 中 显示 。 
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1.2.2 ”创建 和 使 用 母 版 页 


创建 母 版 页 和 使 用 母 版 页 创建 内 容 页 的 方法 与 创建 普通 的 ASPNET 页 面 基本 相同 , 下 
面 通过 一 个 例子 来 具体 说 明 。 

【 例 1-S】 创建 母 版 页 和 内 容 页 。 

本 例 模仿 滨州 学 院 网 站 创建 了 一 个 学 校 网 站 的 母 版 ， 并 从 此 母 版 创建 了 一 个 内 容 页 。 

(1) 创建 一 个 ASPNET Web 应 用 程序 ， 重 命名 为 SchoolWebSite。 

(2) 从 菜单 中 选择 “项 目 ”|“ 添 加 新 项 ”命令 ， 在 弹出 的 “添加 新 项 ”对 话 框 中 选择 
“ 母 版 页 ”， 添 加 一 个 母 版 页 。 

(3) 模拟 滨州 学 院 网 站 首页 ， 在 项 目 中 添加 必要 的 图 片 ， 在 母 版 页 中 添加 一 些 页 面 顶 
部 的 图 片 和 页 面 底部 的 文字 ， 代 码 如 下 : 


<body> 

<form id="form]l" runat="server"> 

<div> 

<! 一 -下 面 的 div 是 母 版 中 的 内 容 ， 页 项 部 的 图 片 和 链接 --> 

<div> 

<img src="images/bzxy.png”alt=" 滨 州 学 院 " /> <br /> <$-- 学 校 LOGO--%> 


< 一 -以 下 各 个 a 标记 是 到 学 校 各 个 部 门 的 链接 --%> 

<a href="http://www.bzu.edu.cn/xuexiaogaikuang/20070821/59.html" class= 
"nav"> 学 校 概况 </a>1 
<a href="http://www.bzu.edu.cn/jiaoxuekeyan/20070830/222.html" class= 
"nav"> 教 学 科研 </a>1 

<a href="http://rsc.bzu.edu.cn/"” class="nav"> 师 资 队伍 </a>| 
<a href="http://zs.bzu.edu.cn/" target=" blank" class="nav"> 招 生 就 业 </a>| 
<a href="http://www.bzu.edu.cn/guojijiaoliu/20070830/233.html" class= 
"nav"> 国 际 交 流 </a>1 
<a href="http://www.bzu.edu.cn/yuanxishezhi/20070913/349.html" class= 
"nav"> 院 系 设置 </a>1| 
<a href="http://www.bzu.edu.cn/dangzhengjigou/20070830/237.html" class= 
"nav"> 党 政 机 构 </a>1 
<a href="http://www.bzu.edu.cn/shetuanjigou/20070830/250.html" class= 
"nav"> 群 团 机 构 </a>1 

<a href="http://hqgl.bzu.edu.cn/" target=" blank" class="nav"> 后 勤 服务 </a> 
</div> 

<asp:ContentPlaceHolder ID="ContentPlaceHolderl" runat="server"> 
<!-- 这 里 是 占 位 符 ， 内 容 页 可 以 修改 这 里 的 内 容 。--> 

</asp:ContentPlaceHolder> 

<!-- 下 面 的 div 是 母 版 中 的 内 容 ， 页 脚 ， 显 示 了 联系 方式 、 版 权 信 息 、 网 站 备案 --> 
<div><table align="center" border="0" cellpadding="0" cellspacing="0" 
width="931"> 

<tr><td class="end" background="/foot.jpg" height="80"> 

<div align="center"> 

<p> 版 权 所 有 &#169; ”滨州 学 院 地 址 : 山东 省 滨州 市 黄河 五 路 391 号 邮编 : 256600 电话 : 
0543-3190016 <br> 

网 站 管理 : 滨州 学 院 网 络 中 心 

<a href="http://www.miibeian.gov.cn/"” target=" blank"> 鲁 ICP 备 07012503 号 
</a> 

<a href="http://www-bzga-gov-cn/" target=" blank"> 滨 公 备 0701173 号 </a> 

<a href="http://www-cyberpolice.cn/"” target=" blank"> 网 络 报警 </a></p> 
> </Er> 
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</table> </div> 
</div> 

</form> 

</body> 


(4) 从 菜单 中 选择 “项 目 ”|“ 添 加 新 项 ”命令 ， 在 弹出 的 “添加 新 项 ”对 话 框 中 选择 
“使 用 母 版 页 的 Web 窗 体 ”， 如 图 1.9 所 示 。 


旱 Xx| 
[ 搜索 己 安装 的 模 站 | 
划 类 型 : Visual C# 
用 于 根据 母 版 页 生成 Web 应 用 程序 的 窗 体 


Reporting 大 类 Visusl C# 
a 回 wr Visuu Cg 
加 eemsw inaa c# 
号 IL 页 Visual C# 
得 本 下 Visud C8 
晶 | Tcrint 文件 Visual C# 

3 ASP_ HET 处 理 程序 visou c# | 


图 1.9 添加 Web 内 容 窗 体 


(5) 在 图 1.9 所 示 的 对 话 框 中 选择 了 “使 用 母 版 页 的 Web 窗 体 ”并 单 击 “ 添 加 ”按钮 
以 后 ,会 出 现 如 图 1.10 所 示 的 对 话 框 ， 要 求 选择 一 个 母 版 页 。 选 择 上 一 步骤 中 所 创建 的 母 
版 页 ， 单 击 “确定 ” 按 钮 。 

EECEEER 3 


项 目 文件 夹 下 ): 文件 夹 内 容 CC): 
日 sitel. Master 


SMpp_Data 
由 国 inages 

由 创 Properties 
自信 引用 


图 1.10 选择 母 版 页 


(6) 在 创建 的 内 容 页 中 ， 简 单 添加 一 些 说 明文 字 ， 代 码 如 下 : 


<s%-- 下 面 的 aPage 语句 中 的 MasterPageFile 指定 了 页 面 所 使 用 的 母 版 页 --g> 

<s@ Page Title="" Language="C#" MasterPageFile="~/Sitel .Master" 
AutoEventWireup="true" CodeBehind="WebForm]l .aspx.cs" Inherits="SchoolWeb. 
WebForml" %> 


<s-- 下 面 这 个 Content 控件 对 应 母 版 页 中 位 于 head 标 记 中 的 ContentPlaceHolder 控件 -- 名 > 


LL 
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<asp:Content ID="Contentl" ContentPlaceHolderID="head" runat="server"> 
</asp:Content> 
<g-- 下 面 这 个 Content 控件 对 应 母 版 页 中 位 于 body 标 记 中 的 ContentPlaceHolder 控件 --%> 


<asp:Content ID="Content2" ContentPlaceHolderID="ContentPlaceHolder1l" 
runat="server"> 


<h1> 欢 迎 来 到 滨州 学 院 </h1> 


<h3> 这 是 内 容 页 里 的 内 容 </h3> 
</asp:Content> 


(7) 在 浏览 嚣 中 查看 内 容 页 ， 运 行 界面 如 图 1.11 所 示 。 


rr CT Er 历史 G) 书签 外 工具) 帮助 中 
/oes 169 /Todorel aspx 月 


oo 相当 -+ 可- 时 - M- 安 - 口 - 少 %.@- 
1 http:7Alecalhes~-/gebForal aspz | ~ | 上 


ee ; 
AE 
仿 却 / 
Binzhou University 


学 校 概况 | 教学 科研 | 师资 队伍 | 招生 就 业 | 国际 交流 | 崇 系 设置 | 党 政 机 构 | 群 团 机 构 | 后 台 服 务 


欢迎 来 到 滨州 学 院 


这 是 内 容 页 里 的 内 容 


版 权 所 有 滨州 学 院 地 址 ;山东 省 滨州 市 黄河 五 路 391 号 邮编 ，256600 电话 ，0543-3190016 
网 站 管理 滨州 学 院 网 络 中 心 鲁 ICP 备 07012503 号 滨 公 备 0701173 号 网 络 报警 


图 1.11 内 容 页 运行 界面 


1.2.3 ”将 现 有 页 面 转换 为 母 版 页 或 内 容 页 


在 网 站 或 者 Web 应 用 程序 开发 过 程 中 ， 有 时 需要 把 一 个 已 经 存在 的 普通 ASPNET 页 
面 修改 为 母 版 页 或 者 内 容 页 。 把 一 个 现 有 页 面 做 成 母 版 页 的 主要 作用 是 可 以 复 用 此 页 面 中 
的 元 素 ， 而 把 现 有 页 面 做 成 内 容 页 可 以 使 此 页 面具 有 与 母 版 相同 的 布局 ， 从 而 实现 网 站 风 
格 的 标准 化 和 一 致 性 。 这 两 种 转换 都 是 通过 修改 页 面 的 HTML 代码 〈 即 .master 或 者 .aspx 
文件 里 的 代码 ) 实现 的 。 本 节 将 介绍 如 何 实现 这 两 种 转换 。 

仔细 观察 母 版 页 、 内 容 页 与 普通 页 面 的 HIML 代码 可 以 发 现 , 三 者 第 一 行 代 码 是 不 一 
样 的 。 下 面 分 别 列 出 了 普通 页 面 、 母 版 页 、 内 容 页 的 第 一 行 代 码 。 

// 普 通 页 面 OrdinaryPage .aspx 第 一 行 代码 

<%@ Page Language="C#" AutoEventWireup="true" CodeBehind="OrdinaryPage. 

aspx.cs" Inherits="MasterPageSample.OrdinaryPage" $%> 

// 母 版 页 MyMaster .master 第 一 行 代码 


<%@ Master Language="C#" AutoEventWireup="true" CodeBehind="MyMaster. 
master.cs" Inherits="MasterPageSample.MyMaster" %> 


// 内 容 页 ContentPage .aspx 第 一 行 代码 
<%@ Page Title="" Language="C#" MasterPageFile= "~/MyMaster.Master" 


第 1 章 ASPNET 网 络 开发 基础 


AutoEventWireup ="true" CodeBehind= "ContentPage.aspx.cs" Inherits= 
"MasterPageSample.ContentPage" %> 


从 上 述 代码 可 以 看 出 ， 母 版 页 的 master 文件 以 <%@Master 开头 ， 普 通 页 面 和 内 容 页 
面 的 aspx 文件 都 是 以 <%@Page 开头 ， 但 是 内 容 页 面 有 一 个 MasterPageFile 标记 ， 表 示 此 
内 容 页 面 所 使 用 的 是 母 版 页 , 而 普通 的 aspx 页 面 没有 这 个 标记 。 从 这 第 一 行 代码 就 可 以 判 
断 出 来 一 个 页 面 是 母 版 页 、 内 容 页 还 是 普通 页 面 。 

再 注意 查看 三 个 页 面 文件 的 其 余 内 容 可 以 发 现 ， 母 版 页 中 通常 有 两 个 内 容 占 位 符 
ContentPlacerHolder， 每 一 个 占 位 符 都 有 一 个 唯一 DD， 正 是 这 两 个 内 容 占 位 符 为 内 容 页 面 
提供 了 定制 页 面 的 机 制 。 下 面 是 一 个 空白 的 母 版 页 的 代码 〈 仅 包含 两 个 内 容 占 位 符 )。 


<sQ@ Master Language="C#" RutoEventWireup="true"” CodeBehind="MyMaster. 
master.cs" Inherits="MasterPageSample.MyMaster" $%> 
<html xmlns="http://www.w3.0org/1999/xhtml"> 
<head runat="server"> 
<title></title> 
<!-- 这 是 第 一 个 内 容 占 位 符 ， 出 现在 HTML 文档 head 部 分 --> 
<asp:ContentPlaceHolder ID="head" runat="server"> 
</asp:ContentPlaceHolder> 
</head> 
<body> 
<form id="forml" runat="server"> 
<div> 
<!-- 这 是 第 二 个 内 容 占 位 符 ， 出 现在 HTML 文档 body 部 分 --> 
<asp:ContentPlaceHolder ID="ContentPlaceHolder]1l" runat="server"> 
</asp:ContentPlaceHolder> 
</div> 
</form> 
</body> 
</html> 


全 提示 : 一 个 母 版 页 并 非 固定 只 能 包含 两 个 内 容 占 位 符 ， 而 是 可 以 包含 一 到 多 个 占 位 符 。 
这 些 占 位 符 可 以 放 在 HTML 文档 的 任意 部 分 。 开 发 人 员 可 以 根据 需要 灵活 确定 
内 容 占 位 符 的 数量 及 位 置 。 


ASPNET 的 内 容 页 是 在 母 版 页 的 基础 上 新 增加 的 内 容 。 在 内 容 页 面 中 ， 没 有 HIML、 
BODY、HEAD 等 标记 ， 这 些 标记 在 一 个 HTML 文档 里 只 出 现 一 次 ， 已 经 包含 在 母 版 页 里 
了 。 内 容 页 通常 包含 与 母 版 页 相对 应 的 内 容 (Content) 控件 ， 一 个 内 容 控件 对 应 于 母 版 页 
里 一 个 内 容 占 位 符 控件 ， 这 个 对 应 关系 是 由 内 容 控 件 的 ContentPlaceHolderID 属性 确定 的 。 
当 显 示 一 个 内 容 页 时 ，Content 控件 中 的 内 容 就 会 插入 到 母 版 页 中 对 应 的 
ContentPlaceHolder 控件 所 在 的 位 置 ， 从 而 组 成 一 个 完整 的 页 面 。 下 面 是 一 个 空白 内 容 页 的 
代码 。 

<%@ Page Title="" Language="C#" MasterPageFile="~/MYMaster -Master" 


RutoEventWireup="true" CodeBehind="ContentPage.aspx.cs" Inherits= 
"MasterPageSample.ContentPage" %> 

<! 一 第 一 个 内 容 控件 ， 对 应 于 母 版 里 的 第 一 个 内 容 占 位 符 (id 为 head) --> 

<asp:Content ID="Contentl" ContentPlaceHolderID="head" runat="server"> 
</asp:Content> 

<! 一 -第 二 个 内 容 控 件 ， 对 应 于 母 版 里 的 第 二 个 内 容 占 位 符 (id 为 ContentPlaceHolder1) --> 
<asp:Content ID="Content2" ContentPlaceHolderID="ContentPlaceHolderl" 
ruUunat="Server”> 

</asp:Content> 
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全 提示 : 母 版 页 中 包含 ContentPlaceHolder 控件 ， 内 容 页 包含 Content 控件 。 内 容 页 中 的 
Content 控件 与 母 版 页 中 的 ContentPlaceHolder 控件 一 一 对 应 , 这 种 对 应 关系 通过 
Conent 控件 的 ContentPlaceHolderID 属性 确定 。 


前 面 分 析 了 母 版 页 、 内 容 页 中 HTML 代码 的 含义 及 两 者 对 应 关系 ,根据 这 些 知识 ， 可 
以 容易 地 把 一 个 普通 ASPX 页 面 转换 成 母 版 页 、 内 容 页 ， 或 者 做 相反 的 转换 。 下 面 通 过 两 
个 例子 做 具体 说 明 。 

【 例 1-6】 从 普通 页 面 提取 母 版 页 。 

本 例 演 示 如 何 基于 一 个 现 有 的 普通 ASPX 页 面 提 取出 母 版 页 。 

(1) 创建 一 个 ASPNET Web 应 用 程序 或 者 网 站 。 

(2) 在 网 站 中 添加 一 个 普通 ASPNET 页 面 ， 重 命名 为 OrdinaryPage.aspx。 在 页 面 中 加 
入 如 下 HTML 代码 。 


<body> 

<form id="forml" runat="server"> 
<div> <s-- 页 面 最 外 层 div--%> 
<div> <s-- 页 面 项 部 div--%> 


<h3> 这 里 放 页 面 顶部 的 flash， 大 横幅 广告 等 </h3> 
<h3> 这 里 放置 页 面部 分 链接 </h3> 
</div> 
<div style=" border:solid 2px silver; height:200px;"> <h3> 这 里 是 页 面 的 具 
体内 容 </h3></div> 
<div> 这 是 页 脚 </div> 
</div> 
</form> 
</body> 


分 析 上 述 页 面 的 代码 , 可 以 看 到 大 体 可 以 分 为 三 部 分 , 页 面 顶部 的 flash 和 链接 是 一 部 
分 ， 页 面 中 间 的 正文 内 容 是 一 部 分 ， 页 脚 内 容 是 一 部 分 。 这 三 部 分 内 容 中 ， 页 头 和 页 脚 内 
容 对 于 所 有 页 面 是 相同 的 。 为 了 优化 网 站 的 设计 ， 可 以 把 页 顶 和 页 脚 放 到 母 版 页 中 。 

(3) 在 项 目 中 添加 一 个 母 版 页 ， 重 命名 为 MyMastermaster。 修 改 母 版 页 的 body 部 分 ， 
在 内 容 占 位 符 的 前 面 加 入 页 头 ， 在 内 容 占 位 符 之 后 加 入 页 脚 。 修 改 后 的 母 版 页 代码 如 下 : 


<body> 
<form id="forml" runat="server"> 
<div> <s%-- 页 面 最 外 层 div--%> 
<div> <s-- 页 面 顶部 div--%> 


<h3> 这 里 放 页 面 顶部 的 ELlash， 大 横幅 广告 等 </h3> 
<h3> 这 里 放置 页 面部 分 链接 </h3> 
</div> 
<asp:ContentPlaceHolder ID="ContentPlaceHolder1"” runat="server"> 
</asp:ContentPlaceHolder> <s-- 母 版 页 的 内 容 占 位 符 控件 --s> 
<div> 这 是 页 脚 </div> <%-- 页 脚 div--%> 
</div> 
</form> 
</body> 


【 例 1-7】 将 普通 页 面 转换 为 内 容 页 。 
本 例 演示 如 何 将 一 个 现 有 的 普通 ASPX 页 面 转换 为 一 个 内 容 页 ， 即 使 其 建立 在 一 个 母 
版 页 的 基础 上 。 
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(1) 创建 一 个 ASPNET Web 应 用 程序 。 
(2) 在 项 目 中 添加 一 个 新 页 面 OrdinaryPage.aspx， 修 改 页 面 代码 如 下 : 


<head id="Headl" runat="server"> 
<title> 这 是 我 的 页 面 </title> 
<script type="text/javascript"> alert (' 这 是 一 段 javascript 代码 '); </script> 
</head> 
<body> 
<form id="forml" runat="server"> 
<div style="” border:solid 2px silver; height:200px;"> <h3> 这 是 页 面 的 内 容 
</h3></div> 
</form> 
</body> 
</html> 


从 上 述 代码 可 以 看 出 ， 这 是 一 个 独立 的 ASPX 页 面 ， 页 面 标题 为 “这 是 我 的 页 面 ” 
页 面 中 一 行文 字 。 如 果 要 修改 此 页 面 的 外 观 与 其 他 页 面 一 致 ， 给 此 页 面 加 上 与 其 他 页 面 一 
样 的 页 头 、 页 脚 ， 那 么 最 合理 也 是 简单 的 方法 就 是 使 用 母 版 页 ， 让 此 页 面 成 为 母 版 页 的 一 
个 内 容 页 面 。 

(3) 把 例 1-6 中 的 母 版 页 MyMastermaster 复制 到 本 项 目 中 。 为 节省 篇 幅 ， 此 处 不 重复 
列 出 母 版 页 的 代码 。 

(4) 修 改 OrdinaryPage.aspx 文件 , 在 第 一 行 代码 <%@Page 的 后 面 添加 MasterPageFile= 
“~/MyMaster.Master"， 表 示 此 页 面 是 基于 MyMaster 母 版 页 创建 的 一 个 内 容 页 ， 再 添加 
“Title=" 这 是 我 的 网 站 ” 表示 内 容 页 的 标题 是 “这 是 我 的 网 站 ”。 修 改 后 的 代码 如 下 : 

<%@ Page MasterPageFile="~/MyMaster.Master" Title=" 这 是 我 的 网 站 " Language= 


"C#" AutogventWireup="true" CodeBehind="OrdinaryPage.aspx.cs" Inherits= 
"MasterPageSample.OrdinaryPage" 和 > 


(5) 由 于 内 容 页 中 没有 HIML、BODY、HEAD 等 标记 ， 所 以 在 OrdinaryPage.aspx 中 
删除 这 些 标 记 。 注 意 删除 的 时 候 不 能 删除 页 面 内 容 。 删 除 这 些 标 记 后 的 页 面 代码 如 下 : 
<gQ@ Page MasterPageFile="~/MyMaster.Master"” Title=" 这 是 我 的 网 站 " Language= 


"C#" RutoEventWireup="true" CodeBehind="OrdinaryPage.aspx.cs" Inherits= 
"MasterPageSample.OrdinaryPage" 多 > 


<script type="text/javascript"> alert(' 这 是 一 段 javascript 代码 '); 
</script> 

<div style=" border:solid 2px silver; height:200px;"> <h3> 这 是 页 面 的 内 容 
</h3></div> 


(6) 在 内 容 页 OrdinaryPage 中 添加 Content 控件 。 如 前 所 述 ， 内 容 页 的 Content 控件 与 
母 版 页 的 ContentPlaceHolder 控件 是 一 一 对 应 的 ， 所 以 ， 根 据 母 版 页 中 ContentPlaceHolder 
控件 的 多 少 及 其 ID 可 以 确定 内 容 页 中 Content 控件 的 个 数 和 属性 。 代 码 如 下 : 


<asp:Content ID="Contentl" ContentPlaceHolderID="head" runat="server"> 
</asp:Content> 
<asp:Content ID="Content2" ContentPlaceHolderID="ContentPlaceHolderl" 
runat="server"> 


</asp:Content> 


(7) 把 原 有 内 容 页 中 的 网 页 内 容 移动 到 相应 的 Content 控件 中 。 在 本 例 中 ， 把 原 有 页 
面 head 标记 中 的 内 容 移动 到 ContentPlaceHolderID 为 head 的 Content 控件 中 ， 而 把 页 面 正 


。17。 


第 1 篇 ASPNET 网 络 开发 关键 技术 


文 内 容 移 动 到 ContentPlaceHolderID 为 ContentPlaceHolderl 的 Content 控件 中 。 完 成 后 整个 
页 面 代码 如 下 : 
<%@ Page MasterPageFile="~/MyMaster .Master" Title=" 这 是 我 的 网 站 " Language= 
"C#" RutoEventWireup="true"” CodeBehind="OrdinaryPage.aspx.cs" Inherits= 
"MasterPageSample.OrdinaryPage" %> 
<asp:Content ID="Contentl" ContentPlaceHolderID="head" runat="server"> 
<script type="text/javascript"> alert(' 这 是 一 段 javascript 代码 '); 
</script> 
</asp:Content> 
<asp:Content ID="Content2" ContentPlaceHolderID="ContentPlaceHolder1l" 
runat="server"> 
<div style="” border:solid 2px silver; height:200px;"> <h3> 这 是 页 面 的 内 容 
</h3></div> 
</asp:Content> 


1.2.4 访 套 母 版 页 


基于 母 版 页 不 但 可 以 创建 出 内 容 页， 而 且 还 可 以 创建 出 其 他 母 版 页 。 这 种 基于 母 版 页 
创建 的 母 版 页 称 为 撕 套 母 版 页 。 母 版 页 可 以 嵌 套 多 层 ， 但 是 从 实用 的 角度 来 说 ， 母 版 页 撕 
套 一 般 不 超过 两 层 。 

母 版 页 典 套 是 一 种 非常 实用 的 机 制 。 例 如 ， 某 网 站 有 许多 页 面 ， 这 些 页 面 按照 外 观 布 
局 大 致 可 以 分 为 两 大 类 ， 第 一 类 页 面 由 纵向 排列 的 页 头 、 内 容 、 页 脚 三 部 分 内 容 组 成 ， 第 
二 类 页 面 与 第 一 类 页 面 不 同 之 处 在 于 中 间 的 内 容 部 分 又 拆 分 成 为 两 部 分 : 左 侧 导航 栏 和 页 
面 内 容 。 这 两 大 类 页 面 的 页 头 和 页 脚 都 是 相同 的 。 利 用 母 版 页 入 套 就 很 容易 设计 出 合理 的 
母 版 和 页 面 。 下 面 通过 一 个 例子 具体 说 明 这 种 设计 方法 。 

【 例 1-8】 堪 套 母 版 页 。 

本 例 演 示 媒 套 母 版 页 的 使 用 。 网 站 中 有 两 个 母 版 页 ， 第 一 个 母 版 页 包含 页 头 、 内 容 占 
位 符 和 页 脚 。 第 二 个 母 版 页 是 一 个 嵌 套 母 版 页 ， 在 第 一 个 母 版 页 的 基础 上 ， 把 第 一 个 母 版 
页 的 内 容 占 位 符 拆 分 成 了 左右 两 部 分 ， 左 边 是 导航 栏 ， 右 边 是 内 容 占 位 符 。 

利用 媒 套 母 版 页 创建 两 类 既 有 联系 又 有 区 别 的 页 面 。 

(1) 创建 一 个 ASPNET Web 应 用 程序 。 

(2) 在 项 目 中 添加 一 个 母 版 页 Masterl.master， 这 个 母 版 页 包含 页 头 、 内 容 占 位 符 和 页 
脚 三 部 分 。 为 了 让 各 个 部 分 边界 更 加 清晰 ， 页 面 中 使 用 了 一 个 CSS 样式 。 页 面 代码 如 下 : 


<head runat="server"> 


<style type="text/css"> <s-- 页 面 中 用 到 的 CSS 样式 --%> 
div { border: solid 2px silver; margin:10px;} 
</style> 


<asp:ContentPlaceHolder ID="head" runat="server"> 
<s--head 中 的 占 位 符 --%> 
</asp:ContentPlaceHolder> 
</head> 
<body> 
<form id="forml" runat="server"> 
<div> 
<div><h3> 第 一 个 母 版 页 HEADER</h3></div> 
<s-- 页 面 body 中 的 内 容 占 位 符 控件 --%> 


<asp:ContentPlaceHolder ID="ContentPlaceHolder1"” runat="server"> 
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</asp:ContentPlaceHolder> 
<div> 
第 一 个 母 版 页 FOOTER 
</div> 
</div> 
</form> 
</body> 
</html> 


Masterl.master 页 面 在 设计 状态 下 的 外 观 如 图 1.12 所 示 。 


第 一 个 母 版 页 HEADER 


,ContentPlaceHolder1 ( 自 定义 
了 第 一 个 母 版 页 FOOTER 


图 1.12 第 一 个 母 版 页 


(3) 从 Visual Studio 菜单 中 选择 “项 目 ”|“ 添 加 新 项 ”命令 ， 在 打开 的 “添加 新 项 ” 
对 话 框 中 选择 “ 典 套 的 母 版 页 ” 将 新 的 母 版 页 命名 为 Master2.master。 在 随后 出 现 的 “ 选 
择 母 版 页 ”对 话 框 中 ， 选 择 第 〈2) 步 所 创建 的 Master1。 所 生成 的 霸 套 母 版 页 代码 如 下 : 

<%@ Master Language="C#" MasterPageFile="~/NestedMaster/Masterl.Master" 

AutoEventWireup="true" CodeBehind= "NestedMasterPagel. master.cs" 

Inherits= "MasterPageSample.NestedMaster. Master2" $%> 

<asp:Content ID="Contentl" ContentPlaceHolderID="head" runat="server"> 

</asp:Content> 

<asp:Content ID="Content2" ContentPlaceHolderID="ContentPlaceHolderl" 

runat="server"> 

</asp:Content> 


从 上 面 这 段 代 码 可 以 看 出 ， 自 动 生 成 的 柑 套 母 版 页 的 代码 几乎 与 自动 生成 的 内 容 代码 
是 相同 的 ， 也 是 包含 两 个 Content 控件 ， 这 两 个 Content 控件 对 应 于 母 版 Masterl 中 的 两 个 
ContentPlaceHolder。 但 是 嵌 套 母 版 页 的 第 一 行 代码 与 内 容 页 不 同 。 母 版 页 以 “<%G@Master” 


开头 ， 而 内 容 页 以 “<%@Page” 开 头 ， 另 外 ， 内 容 页 有 一 个 Title 属性 ， 而 母 版 页 则 没有 
这 个 属性 。 


现在 这 个 母 版 页 其 实 还 算 不 上 一 个 真正 的 母 版 页 ， 因 为 还 不 包含 ContentPlaceHolder 
控件 。 下 面 的 步骤 就 来 放置 ContentPlaceHolder 控件 。 
(4) 在 Master2.master 母 版 页 中 的 第 一 个 Content 控件 中 放置 一 个 ContentPlaceHolder 
控件 ， 代 码 如 下 : 


<asp:Content ID="Contentl" ContentPlaceHolderID="head" runat="server"> 
<asp:ContentPlaceHolder ID="head2" runat="server"> 
</asp:ContentPlaceHolder> 
</asp:Content> 
要 注意 区 分 上 面 这 段 代 码 中 的 Content、ContentPlaceHolder 及 其 关系 。 
(5) 在 Master2.master 母 版 页 中 的 第 二 个 Content 控件 中 放置 两 个 div 标记 , 第 一 个 div 
标记 中 包含 左 侧 导 航 栏 ， 第 二 个 div 中 包含 一 个 ConentPlaceHolder 控件 ， 为 内 容 页 留 出 存 
放 页 面 内 容 的 空间 。 
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<asp:Content ID="Content2" ContentPlaceHolderID="ContentPlaceHolder1" 
runat="server"> 
<div style="float:left;"> 
<h3> 第 二 个 母 版 页 <br /> 左 侧 导 航 栏 </h3> 
</div> 
<div style="float:left; "> 
<asp:ContentPlaceHolder ID="ContentPlaceHolder]1" runat="server"> 
</asp:ContentPlaceHolder> 
</div> 
</asp:Content> 


Master2.master 页 面 在 设计 状态 下 的 外 观 如 图 1.13 所 示 。 
(6) 在 项 目 中 添加 一 个 基于 母 版 页 Master2 的 内 容 页 Master2Page.aspx， 并 在 其 中 写 入 


些 文字 ， 代 码 如 下 : 


第 一 个 母 版 页 HEADER 


pr derl 
左 而 导航 栏 各 
草 一 个 哥 版 页 FOOTER 


名 一 个 母 版 页 FOOTER. 


<%@ Page Title="" Language="C#" MasterPageFile="~/NestedMaster/Master2. 
master" AutoEventWireup="true" CodeBehind="Master2Page.aspx.cs" Inherits= 
"MasterPageSample.NestedMaster.Master2Page" $%> 

<asp:Content ID="Contentl" ContentPlaceHolderID="head2" runat="server"> 
</asp:Content> 

<asp:Content ID="Content2" ContentPlaceHolderID="ContentPlaceHolder1l" 
runat="server"> 

<h3> 内 容 页 面 2 具体 内 容 </h3> 


</asp:Content> 


(7) 在 浏览 器 中 查看 Master2Page.aspx 页 面 ， 效 果 如 图 1.14 所 示 。 


Er rrr ET 历史 人 @) 书签 如 工具 中 


[C5 (cr -el 
C= J 1» © -@- 


第 一 个 母 版 页 HEADER 


i 第 二 个 母 版 页 “内容 页 面 2 具体 内 容 
第 三 个 母 版 页 | 中 ee 恬 介 和 各 


| 圭 成 


图 1.13 尾 套 母 版 页 图 1.14 内 容 页 外 观 效果 


1.2.5 ”从 内 容 页 访问 母 版 页 控件 


在 ASPNET 开发 环境 下 , 母 版 页 与 内 容 页 是 两 个 不 同 的 页 面 , 存储 在 两 个 不 同 的 文件 


中 ， 内 容 页 不 能 直接 访问 母 版 页 的 控件 。 但 是 在 实际 应 用 中 ， 却 经 常 需要 这 种 访问 ， 这 就 
需要 通过 一 些 编程 技巧 来 实现 。 


ASPNET 页 面 是 System.Web.ULPage 类 的 派生 类 , 母 版 页 是 System.Web.ULMasterPage 


类 的 派生 类 。Page 类 有 一 个 Master 属性 ， 表 示 此 页 面 的 母 版 页 ， 代 码 如 下 : 


public MasterPage Master { get; } 
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通过 Page 类 的 Master 属性 ， 就 可 以 访问 到 页 面 的 母 版 页 。 而 MasterPage 类 继承 自 
UserControl 类 ， 因 此 调用 其 FindControl0 方 法 就 可 以 得 到 母 版 页 上 的 某 个 控件 。 

【 例 1-9】 访问 母 版 页 控件 (方法 1)。 

许多 网 站 的 页 面 上 都 有 一 个 “当前 位 置 ”的 标识 , 用 于 说 明 当 前 正在 访问 的 页 面 名 称 。 
由 于 这 个 标识 存在 于 多 个 页 面 ， 所 以 可 以 把 这 个 标识 放 在 母 版 页 中 。 但 是 对 于 各 个 不 同 页 
面 来 说 ， 这 个 标识 是 不 一 样 的 ， 需 要 在 内 容 页 中 修改 这 个 标识 的 内 容 。 

(1) 创建 一 个 ASPNET Web 应 用 程序 。 

(2) 在 项 目 中 添加 一 个 母 版 页 Sitel1， 并 在 母 版 页 上 放置 一 个 ID 为 currentPosition 的 
Label， 用 于 显示 当前 位 置 。 页 面 主要 代码 如 下 : 

<body> 

<form id="forml" runat="server"> 


<div> 

<div> 

<div><h3> 页 面 项 部 固定 内 容 </h3></div> 

<div> 当 前 位 置 : 
<asp:Label ID="currentPosition" runat="server” Text=" 你 的 位 置 "> 
</asp:Label> </div> 

</div> 
<asp:ContentPlaceHolder ID="ContentPlaceHolder]l" runat="server"> 
</asp:ContentPlaceHolder> 


<div> 页 脚 固定 内 容 </div> 
</div> 
</form> 
</body> 
(3) 在 项 目 中 添加 一 个 基于 母 版 页 Sitel 的 内 容 页 HomePage.aspx。 在 内 容 页 的 Load 


事件 中 ， 编 写 代码 修改 母 版 页 中 Label 控件 的 文本 。 


protected void Page Load (object sender, EventArgs e) 
1 
if (!IsPostBack) 
{ 
// 得 到 母 版 页 上 用 于 标识 当前 位 置 的 Label 控件 


Label position = this.Master.FindControl ("currentPosition") as 
Label; 
// 如 果 找 到 了 这 个 Label 控件 ， 同 时 设置 其 文本 
if ( position!=null) 
position.Text = "<a href='HomePage.aspx'> 首 页 </a>"; 


} 


(4) 运行 HomePage.aspx， 运 行 界面 如 图 1.15 所 示 。 

例 1-9 演示 了 通过 FindControl 访问 母 版 页 控件 的 方法 。 这 种 方法 的 优点 是 简单 、 适 应 
性 强 ， 只 需要 编写 少量 代码 ， 就 可 以 用 于 访问 任何 类 型 控件 。 但 这 种 方法 也 存在 以 下 几 个 
缺点 。 

口 如 果 母 版 页 里 控件 很 多 ， 很 难 记 住 其 中 控件 的 ID。 

口 在 调用 FindControl 时 , 如 果 把 控件 的 ID 写 错 了 , Visual Studio 不 能 给 出 错误 提示 ， 

只 有 当 程序 运行 时 才 会 产生 错误 。 

口 FindControl 方法 返回 Control 类 型 而 非 具 体 类 型 (如 Label) ， 需 要 进行 强制 类 型 

转换 。 
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我 的 网 站 名 称 - merilla Firefez 
文件 四 查看 WD 历史 G) 书签 如 ”工具 帮助 四 
Ee /et - esl ea 
El -1+» 马 -@- 
1 我 的 网 站 名 称 | 下 | 
页 面 项 部 固定 内 容 

当前 位 置 ， 首页 

内 容 页 : 欢迎 访问 我 的 网 站 
页 脚 固定 内 容 
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图 1.15 例 1-9 运 行 界面 


口 从 面向 对 象 的 角度 来 讲 ， 这 种 方式 从 一 个 类 (内容 页 ) 访问 另 一 个 类 《〈 母 版 页 ) 
的 私有 成 员 ， 破 坏 了 母 版 页 的 封装 。 

针对 上 述 缺 点 ， 可 以 用 另外 一 种 方式 访问 母 版 页 控件 。 这 种 方式 的 思路 是 在 母 版 页 中 
声明 一 个 属性 ， 内 容 页 通过 这 个 属性 来 访问 母 版 页 上 的 控件 。 这 样 ， 只 有 母 版 页 本 身 才 与 
其 中 的 控件 打交道 ， 内 容 页 不 需要 知道 母 版 页 上 控件 的 ID， 而 且 属 性 是 强 类 型 的 ， 从 而 很 
好 地 解决 了 以 上 几 个 问题 。 

【 例 1-10】 访问 母 版 页 控件 (方法 2)。 

(1) 创建 一 个 ASPNET 应 用 程序 ， 添 加 一 个 母 版 页 Sitel.master， 其 HTML 代码 与 例 
1-9 完全 相同 。 

(2) 在 Sitel.master.cs 文件 中 ， 为 母 版 页 添加 一 个 属性 myPosition， 用 于 控制 母 版 页 
Label 上 的 文本 。 代 码 如 下 : 

// 将 母 版 页 中 标记 当前 位 置 的 Label 的 文本 封装 成 为 一 个 属性 以 方便 访问 

public string MyPosition 

! get { return currentPosition.Text; } 


set { currentPosition.Text = value; } 


; 


(3) 在 项 目 中 添加 一 个 内 容 页 HomePage.aspx, 并 在 页 面 的 Load 事件 中 设置 当前 位 置 。 
代码 如 下 : 
protected void Page Load(object sender, EventArgs e) 
{ 
if (!IsPostBack) 
{ 
Sitel master = this.Master as Sitel; // 获 得 母 版 页 


master.MyPosition = "首页 "; // 设 置 母 版 上 的 当前 位 置 
上 
} 


1.3 主 题 


ASPNET 的 主题 (Theme) 是 一 种 可 以 集中 控制 网 页 控件 外 观 的 机 制 。 通过 应 用 主题 ， 
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可 以 使 网 站 中 的 控件 具有 一 致 的 外 观 , 例如 , 让 所 有 的 Button 都 具有 相同 的 边框 、 背 景色 、 
字体 等 。ASPNET 的 主题 与 HIML 中 的 CSS 既 有 联系 又 有 区 别 ， 两 者 经 常 配合 使 用 。 


1.3.1 创建 和 使 用 主题 


与 ASPNET 主题 密切 相关 的 一 个 概念 是 外 观 (Skin)。 一 个 主题 确定 了 一 种 页 面 风格 ， 
而 一 个 外 观 则 定义 了 一 种 控件 的 样式 。Skin 的 意思 是 皮肤 ， 这 个 单词 很 明确 地 表明 了 
ASPNET Skin 的 作用 ， 就 是 定义 控件 呈现 出 来 的 样子 。 通 过 将 主题 和 外 观 应 用 于 网 站 或 者 
网 页 ， 就 可 以 定制 该 范围 内 控件 的 样式 。 

在 ASPNET 中 , 一 个 主题 对 应 于 一 个 特殊 的 文件 夹 , 而 一 个 外 观 则 通常 对 应 于 某 主 题 
文件 夹 下 的 一 个 文件 。 一 个 主题 可 以 包含 0 到 多 个 外 观 文件 ， 这 些 外 观 文件 定义 了 各 种 控 
件 在 该 主题 下 所 呈现 出 来 的 样式 。 

下 面 通过 一 个 例子 介绍 如 何 使 用 主题 和 外 观 。 

【 例 1-11】 应 用 主题 和 外 观 。 

本 例 演示 如 何 创建 主题 ， 如 何 添加 外 观 ， 以 及 如 何 将 所 创建 的 主题 和 外 观 应 用 于 网 站 
和 页 面 。 

(1) 创建 一 个 ASPNET Web 应 用 程序 ， 命 名 为 ThemeSample。 

(2) 在 Visual Studio 解决 方案 资源 管理 器 中 ， 右 击 项 目 名 称 ， 从 弹出 的 快捷 菜单 中 选 
择 “ 添 加 ”|“ 添 加 ASPNET 文件 夹 ”| “主题 ”命令 ， 如 图 1.16 所 示 。 


项 目 生成 质 序 区 ) 
司 | 新昌 项 由 
添加 引用 @)，. 国 现 有 项 @). 
添加 Web 引用 到 ) 国 | ae 文件 赤 四 ) 
总 查 看 类 关系 图 App_GlobalResources (R) 
设 为 启动 项 目 &) 闭 | 姐 件 wD.… App_LocslResources CC) 
调试 @) | 局 类 加 .… App A 
苹 将 解 志方 案 于 加 到 源 代码 管理 Q)， war 人 
#0 [ #0 
图 1.16 添加 主题 文件 夹 
(3) 添加 主题 后 ,会 在 项 目 中 增加 两 个 文件 夹 : 一 个 是 App_Theme， 另 外 一 个 是 “ 主 


题 1” 文 件 夹 。 其 中 App_Theme 是 一 个 ASPNET 系统 文件 夹 ， 用 于 存放 网 站 中 的 主题 。 
这 个 文件 夹 只 有 当 第 一 次 添加 主题 时 才 会 创建 。 另 外 一 个 “主题 1” 文 件 夹 就 是 主题 所 对 
应 的 文件 夹 ， 文 件 夹 的 名 称 就 是 主题 的 名 称 。 把 “主题 1” 重 合 名 为 GrassGreen 。 

(4) 在 GrassGreen 主题 文件 夹 上 右 击 ， 从 弹出 的 快捷 菜单 中 选择 “添加 ”|“ 新 建 项 ” 
命令 ， 在 弹出 的 “添加 新 项 ”对 话 框 中 选择 “外 观 文件 ”把 文件 重 命名 为 Sample.skin， 
单 击 “ 添 加 ”按钮 。 

(5) 删除 Sample.skin 文件 中 自动 生成 的 内 容 ， 然 后 添加 以 下 代码 : 


<asp:Button runat="server" BackColor="lightgreen" BorderStyle="Solid" 
BorderColor= "DarkGreen"/> 


二 
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全 注意 : 在 外 观 文件 中 不 能 包含 除 控件 标记 以 外 的 其 他 文字 内 容 ， 包 括 不 能 包含 注释 ， 否 
则 页 面 运行 时 会 提示 分 析 器 错误 。 


上 面 的 代码 设置 了 Button 控件 的 外 观 : 浅 绿色 背景 ， 深 绿色 实 线 边框 。 可 以 看 出 ， 这 
种 设置 外 观 的 语法 与 ASPNET 页 面 中 设置 控件 的 语法 相同 ， 而 与 HTML 中 CSS 的 语法 
不 同 。 


全 提示 : 外 观 文 件 中 通常 包含 很 多 属性 ， 而 且 不 能 直接 看 到 这 些 属性 对 应 的 效果 ， 这 给 外 
观 文件 代码 编写 造成 不 小 的 困难 。 可 以 先 在 ASP.NET 页 面 中 以 所 见 即 所 得 方式 
设置 好 控件 外 观 属 性 , 然后 再 复制 到 外 观 文件 中 。 注意 在 复制 代码 时 要 把 控件 ID 
删除 。 


(6) 前 面 几 个 步骤 创建 了 主题 和 一 个 外 观 文件 ， 下 面 将 使 用 该 主题 和 外 观 。 在 项 目 中 
添加 一 个 页 面 ThemePage.aspx， 在 页 面 上 放 两 个 Button 控件 。 此 时 按钮 的 外 观 仍然 是 默认 
的 灰色 按钮 。 如 果 要 应 用 前 一 步骤 中 所 创建 的 主题 ， 那 么 应 该 修改 一 下 ThemePage.aspx 页 
面 代 码 ， 在 页 面 第 一 行 的 <%@ Page 后 面 添加 一 个 Theme 属性 ， 代 码 如 下 : 

<#sQ@ Page Language="C#" RutoEventWireup="true"” CodeBehind="ThemePage . 

aspx.cs" Inherits="ThemeSample.ThemePage" Theme="GrassGreen" %> 

(7) 在 浏览 器 中 查看 ThemePage 页 面 ， 可 以 看 到 两 个 按钮 都 变 成 了 绿色 背景 和 边框 。 

(8) 在 ThemePage.aspx 页 面 上 添加 一 个 HTML 的 Button， 再 次 运行 页 面 ， 可 以 看 到 
ASPNET 的 Button 变 成 绿色 ， 而 HTML 的 Button 并 没有 改变 颜色 。 这 是 因为 主题 和 外 观 
仅 对 服务 器 端 控件 有 效 。 


外 提示 : ASPNET 主题 和 外 观 仅 对 服务 器 端 控件 有 效 ， 对 于 HTML 客户 端 控件 无 效 


上 一 个 例子 实现 了 主题 、 外 观 的 创建 和 应 用 ， 仔 细 考 虑 这 个 例子 ， 会 发 现 两 个 问题 。 

(1) 如 果 一 个 页 面 有 多 个 Button 控件 ， 有 的 Button 需要 应 用 主题 和 外 观 , 而 有 的 则 不 
需要 ， 应 该 如 何 加 以 控制 。 

(2) 一 个 网 站 中 有 许多 页 面 ， 所 有 页 面 都 需要 应 用 主题 。 如 果 每 个 页 面 修改 <%@Page 
语句 的 话 ， 工 作 量 太 大 。 而 且 将 来 如 果 要 更 改 主题 ， 那 么 修改 的 工作 量 也 相当 大 。 

这 两 个 问题 在 ASPNET 中 都 有 很 好 的 解决 方案 ， 使 用 命名 主题 可 以 解决 第 一 个 问题 
使 用 配置 文件 可 以 解决 第 二 个 问题 。 下 面 通过 一 个 例子 演示 具体 的 代码 实现 。 

【 例 1-12】 命名 外 观 和 配置 文件 。 

(1) 创建 一 个 ASPNET 应 用 程序 。 

(2) 在 项 目 中 添加 一 个 主题 SkyBlue。 在 主题 中 添加 一 个 外 观 文件 Test.skin， 代 码 
如 下 : 


<asp:Button SkinID="skin1" runat="server" BackColor="AliceBlue" 

BorderStyle="Solid" BorderColor="Blue" Font-Bold="true" /> 

<asp:Button SkinID="skin2" runat="server" BackColor="Silver" BorderStyle= 

"Solid" BorderColor="Blue" /> 

在 上 述 外 观 文 件 代 码 中 , 每 个 控件 都 有 一 个 SkinID 属性 , 这 个 属性 指定 了 此 外 观 的 名 
称 ， 这 种 外 观 称 为 命名 外 观 。 在 例 1-11 中 所 创建 的 不 带 SkinID 的 外 观 称 为 默认 外 观 。 如 
果 页 面 使 用 了 主题 ， 则 默认 外 观 会 自动 应 用 到 页 面 上 的 对 应 控件 ， 而 命名 外 观 必须 通过 其 
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名 称 明确 引用 。 


全 注意 : 在 同一 个 主题 中 ， 同 一 种 控件 不 允许 有 重 名 的 命名 外 观 ， 也 不 允许 有 多 于 一 个 默 
认 外 观 。 


Test.skin 文件 中 为 Button 控件 定义 了 两 个 命名 外 观 , 其 ID 分 别 为 skinl 和 skin2。skin1 
定义 的 Button 样式 为 浅 蓝 色 背景 、 蓝 色 实 线 边框 、 粗 体 字 ，skin2 定义 的 Button 样式 为 银 
灰色 背景 、 蓝 色 边 框 。 

(3) 在 项 目 中 添加 一 个 页 面 ThemePage2.aspx， 为 页 面 指定 主题 为 SkyBlue， 页 面 中 包 
含 三 个 Button 控件 ， 页 面 关 键 代码 如 下 : 

<%@ Page Language="C#" AutoEventWireup="true" CodeBehind="ThemePage2. 

aspx.cs" Inherits="ThemeSample.sample2.ThemePage2" Theme="SkyBlue" %> 

<body> 

<form id="forml" runat="server"> 

<div> 
<asp:Button ID="Button1"” runat="server"” Text=" 测 试 按钮 1"/> 
<asp:Button ID="Button2"” runat="server"” Text=" 测 试 按钮 2"/> 
<asp:Button ID="Button3"” runat="server"” Text=" 测 试 按钮 3"/> 

</div> 

</form> 

</body> 

此 时 如 果 浏 览 ThemePage2.aspx 页 面 ， 可 以 看 到 3 个 按钮 都 是 默认 样式 ， 说 明 在 
Test.skin 外 观 文件 中 定义 的 2 个 Button 的 命名 外 观 并 没有 被 应 用 。 

(4) 通过 Button 控件 的 SkinID 属性 为 前 2 个 按钮 分 别 指定 skinl 和 skin2 外 观 ， 而 第 
3 个 按钮 不 使 用 命名 外 观 。 代 码 如 下 : 

<asp:Button ID="Buttonl" runat="server" Text=" 测 试 按钮 1" SkinID="skin1" /> 

<asp:Button ID="Button2" runat="server" Text=" 测 试 按钮 2" SkinID="skin2" /> 

<asp:Button ID="Button3" runat="server"” Text=" 测 试 按钮 3" /> 


(5) 在 浏览 器 中 查看 ThemePage2.aspx， 可 以 看 到 Buttonl 样式 为 浅 蓝 色 背 景 、 蓝 色 实 
线 边 框 、 粗 体 字 ，Button2 样式 为 银灰 色 背 景 、 蓝 色 实 线 边框 ， 说 明 两 个 按钮 已 经 分 别 应 用 
了 skinl 和 skin2 外 观 。ThemePage2 页 面 运行 界面 如 图 1.17 所 示 。 


文件 四 rT 历史 @E) | 书签 @ 工具 I) 帮助 0 
EIB ex | mm - [le 
| +» - 国 - 
Cer 


| 完成 


ES 区 
图 1.17 ThemePage2 运行 界面 


(6) 通常 网 站 中 会 有 多 个 页 面 都 需要 使 用 主题 。 通 过 修改 <%@Page 页 面 指令 来 使 用 
主题 显然 是 一 种 费时 费力 的 笨 办 法 ， 可 以 通过 配置 文件 一 次 性 将 主题 应 用 于 所 有 页 面 。 在 
本 项 目 中 ,打开 Web 配置 文件 web.config (如 果 文 件 不 存在 ， 则 添加 一 个 )， 在 system.web 
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结 点 中 添加 一 个 pages 结 点 ， 并 指定 主题 ， 代 码 如 下 : 


<system.web> 
<pages theme="SkyBlue"/> 
</system.web> 


(7) 打开 ThemePage2.aspx 代码 ， 并 去 掉 <%(@Page 后 面 的 Theme 属性 ， 重 新 编译 运 
行 该 页 面 ， 可 以 看 到 Button 控件 的 外 观 仍 然 有 效 。 这 是 由 于 通过 web.config 中 的 配置 已 经 
把 SkyBlue 主题 应 用 到 了 所 有 页 面 。 如 果 再 添加 新 的 页 面 ， 也 会 自动 应 用 SkyBlue 主题 。 


1.3.2 ”主题 与 样式 表 


主题 与 样式 表 CSS 都 可 用 于 控制 页 面 外 观 。 主 题 只 能 用 于 ASPNET 服务 器 控件 ， 而 
CSS 可 用 于 任意 HTML 元 素 ， 包 括 ASPNET 服务 器 控件 。 由 于 主题 是 专门 针对 ASPNET 
控件 设计 的 ， 所 以 更 能 灵活 、 精 确 地 设置 ASPNET 控件 的 各 种 外 观 属 性 ， 其 语法 也 接近 
C# 代 码 ， 易 于 为 C# 程 序 员 所 掌握 。 在 开发 Web 程序 时 把 主题 和 CSS 结合 使 用 ， 可 以 实现 
二 者 优势 互补 ， 更 好 地 设计 出 美观 的 页 面 。 

要 在 HTML 中 引用 一 个 CSS 文件 ， 通 常 使 用 以 下 语法 ， 其 中 MyCss.css 为 要 引用 的 
CSS 文件 名 。 


<link href="MyCss.css" rel="stylesheet" type="text/css" /> 


当主 题 和 CSS 结合 使 用 时 ， 可 以 把 CSS 文件 放 到 ASPNET 的 主题 文件 夹 中 。 如 果 页 
面 应 用 了 某 主 题 ， 则 主题 文件 夹 中 的 所 有 CSS 文件 都 自动 被 页 面 所 包含 ， 而 不 需要 在 页 面 
中 写 link 语句 。 

【 例 1-13】 主题 文件 夹 中 的 CSS 文件 。 

本 例 演示 主题 文件 夹 中 的 CSS 文件 会 自动 被 应 用 该 主题 的 页 面 所 引用 。 

(1) 创建 一 个 ASPNET 应 用 程序 。 

(2) 在 项 目 中 添加 主题 文件 夹 ， 并 添加 一 个 名 为 CssTheme 的 主题 。 

(3) 在 CssTheme 文件 夹 中 添加 一 个 CSS 文件 MyCss.css， 其 中 包含 以 下 代码 : 

/* 定 义 所 有 input 元 素 默认 样式 为 黄色 背景 ， 蓝 色 边 框 */ 


input 
{ 
background-color:Yellow; /* 黄 色 背 景色 */ 
border:solid 1px BLUE; /* 边 框 : 实 线 、 蓝 色 、1 像素 宽 */ 
} 
/* 定 义 一 个 名 为 button 的 Css 类 ， 银 灰色 背景 ， 黑 色 边 框 */ 
-button 
{ 
background-color:silver; /* 银 灰色 背景 色 */ 
border:solid lpx black; /* 边 框 : 实绩 、 黑 色 、1 像素 宽 */ 
. 


(4) 在 项 目 中 添加 一 个 页 面 CssThemePage.aspx， 设 置 页 面 的 主题 为 CssTheme， 并 在 
页 面 上 放置 一 个 TextBox 和 一 个 Button 控件 ， 关 键 代码 如 下 : 


<%@ Page Language="C#" AutoEventWireup="true" CodeBehind="CssThemePage. 
aspx.cs" Inherits="ThemeSample.CssTheme.CssThemePage" Theme="CssTheme" 当 > 
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<body> 
<form id="forml" runat="server"> 
<div> 
<asp:TextBox ID="TextBoxl" runat="server"></asp:TextBox> 
<asp:Button ID="Buttonl" runat="server" Text="Button" CssClass= 
"batton” /> 
</div> 
</form> 
</body> 
</html> 
(5) 运行 此 页 面 ， 可 以 看 到 页 面 上 的 TextBox 控件 和 Button 控件 都 应 用 了 相应 的 CSS 
样式 。 从 浏览 器 菜单 中 选择 “查看 源码 ”命令 ， 可 以 看 到 所 生成 页 面 的 HIML 代码 中 包含 
以 下 内 容 : 
<html xmlns="http://www.w3.0rg/1999/xhtml" > 
<head> 
<title></title> 
<link href="App_ Themes/CssTheme/MyCss.css" type="text/css" rel= 
"stylesheet" /> 
</head> 


其 中 的 link 一 行 代码 引用 了 CssTheme 主题 文件 夹 里 的 CSS 文件 。 这 也 证 实 了 包含 在 
页 面 主题 文件 夹 里 的 CSS 文件 会 被 自动 引用 。 

应 用 主题 可 以 批量 指定 ASPNET 服务 器 控件 的 样式 , 如 果 要 单独 设置 某 一 个 控件 的 样 
式 , 通常 是 在 ASPX 页 面 中 为 控件 指定 相应 属性 ， 例 如 下 面 的 代码 把 Button 背景 色 设 置 为 
浅 绿色 。 

<asp:Button ID="Buttonl" runat="server"” Text=" 测试 按钮 1" BackColor= 

"LightGreen" /> 

如 果 对 于 同一 控件 , 既 应 用 了 主题 和 外 观 , 又 在 ASPX 页 面 中 为 控件 定义 了 内 联 样式 ， 
那么 控件 在 显示 时 会 出 现 什么 结果 呢 ? 如 果 出 现 这 种 情况 , ASPNET 会 把 这 主题 的 样式 和 
页 面 中 的 内 联 样式 合并 起 来 ， 并 按照 一 定 的 优先 级 解决 两 者 的 冲突 。 如 果 在 配置 文件 或 者 
页 面 <%@Page 指定 中 使 用 了 Theme 关键 字 ， 那 么 这 种 主题 的 优先 级 则 高 于 页 面 中 内 联 样 
式 。 如 果 要 想 让 页 面 中 的 内 联 样式 优先 级 高 于 主题 样式 ， 则 在 应 用 主题 时 应 该 使 用 
StyleSheetTheme 关键 字 。 


名 提示: Theme、 控 件 内 联 样式 、StyleSheetTheme 三 者 的 优先 级 依次 降低 。 


【 例 1-14】 Theme 和 StyleSheetTheme。 

本 例 演示 Theme、StyleSheetTheme、 控 件 内 联 样式 的 合并 及 各 自 的 优先 级 。 

(1) 创建 一 个 ASPNET Web 应 用 程序 。 

(2) 在 项 目 中 添加 一 个 GrassGreen 的 主题 ， 并 在 该 主题 中 添加 一 个 skin 文件 。skin 文 
件 为 Button 控件 定义 了 默认 外 观 〈 浅 绿色 背景 色 、 深 绿色 边框 )， 代 码 如 下 : 


<asp:Button runat="server" BackColor="1ightgreen" BorderStyle="Solid" 
BorderColor="DarkGreen" /> 


(3) 在 项 目 中 添加 一 个 页 面 Pagel.aspx， 设 置 该 页 面 主题 为 GrassGreen， 在 页 面 上 放 
置 一 个 Button， 并 设置 Button 的 内 联 样式 。 页 面 关键 代码 如 下 : 
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<%@ Page Language="C#" AutoEventWireup="true" CodeBehind="Pagel.aspx.cs" 

Inherits="ThemeSample.StyleSheetTheme.Pagel" Theme="GrassGreen"%®> 

<body> 

<form id="forml" runat="server"> 
<div> 
<asp:Button ID="Buttonl" runat="server"” Text=" 按 钮 一 " BackColor="White" 
Font-Bold="true" /> 
</div> 
</form> 

</body> 

</html> 

(4) 在 浏览 器 中 查看 Pagel 页 面 , 可 以 看 到 Button 样式 为 浅 绿色 背景 、 绿色 实 线 边 框 、 
粗 体 字 。 这 个 样式 是 主题 中 定义 的 外 观 与 Button 内 联 样式 综合 作用 的 结果 。 在 生成 页 面 时 ， 
ASPNET 首先 把 主题 外 观 与 内 联 样式 合并 ， 得 到 如 下 的 样式 : 

BackColor="lightgreen" BorderSstyle="Solid" BorderColor="DarkGreen" 

BackColor="White" Font-Bold="true" 

这 个 样式 中 包含 两 个 BackColor, 第 一 个 BackColor 来 源 于 主题 外 观 ,第 二 个 BackColor 
来 源 于 内 联 样式 。 这 两 个 BackColor 冲突 ， 按 照 优先 级 来 处 理 ， 由 于 Theme 优先 级 高 于 内 
联 样式 ， 所 以 最 终 Button 的 BackColor 为 主题 中 定义 的 LightGreen 。 

(5) 把 Pagel.aspx 页 面 的 <%@Page 页 面 指令 中 Theme 关键 字 改 为 StyleSheetTheme， 
修改 后 的 代码 如 下 : 

<%@ Page Language="C#" RutoEventWireup="true" CodeBehind="Pagel.aspx.cs" 

Inherits="ThemeSample.StyleSheetTheme.Pagel" StyleSheetTheme="GrassGreen"%> 

(6) 再 次 浏览 Pagel.aspx 页 面 ， 可 以 看 到 Button 的 样式 为 白色 背景 、 绿 色 实 线 边框 、 
粗 体 字 。 这 个 样式 同样 也 是 主题 与 内 联 样式 综合 作用 的 结果 。 在 生成 页 面 时 ，ASPNET 首 
先 把 两 种 样式 合并 ， 然 后 按照 优先 级 解决 合并 后 得 到 的 样式 中 的 冲突 。 由 于 内 联 样式 优先 
级 高 于 StyleSheetTheme， 所 以 最 终 Button 的 背景 色 为 内 联 样式 中 定义 的 白色 。 


1.3.3 动态 修改 主题 


在 有 的 网 站 中 ,允许 用 户 根据 自己 喜好 更 换 网 站 的 皮肤 , 这 个 功能 利用 ASPNET 的 主 
题 机 制 能 够 很 方便 的 实现 。 在 C# 代 码 中 可 以 通过 Page 类 的 Theme 属性 改变 页 面 的 主题 ， 
代码 如 下 : 

Page .Theme=" 要 设置 的 主题 名 称 "; 


上 面 代码 只 有 一 行 ， 看 似 简单 ， 但 是 要 想 在 实际 应 用 中 发 挥 作用 ， 还 需要 考虑 多 个 因 
素 。 例 如 ， 让 用 户 修改 主题 的 一 种 最 直观 的 思路 是 在 页 面 上 用 一 个 下 拉 列 表 框 列 出 主题 ， 
让 用 户 选 择 其 中 一 个 ， 然 后 单 击 一 个 按钮 ， 在 按钮 的 Click 事件 中 根据 用 户 选择 的 主题 进 
行 修改 .但 是 这 种 思路 并 不 可 行 , 因为 根据 对 ASPNET 页 面 生命 周期 的 分 析 , 当 处 理 Button 
的 Click 事件 时 ， 页 面 主题 已 经 应 用 完毕 ， 此 时 不 可 以 再 修改 主题 。 
主题 必须 在 页 面 初始 化 以 前 (通常 是 在 页 面 的 PreInit〉 的 事件 中 进行 修改 ， 而 此 时 控 
件 的 状态 尚 不 可 用 ， 也 就 无 法 获得 DropDownList 中 所 选择 的 值 。 这 样 就 形成 了 一 种 矛盾 : 
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在 页 面 生命 周期 的 早期 阶段 (如 PreInit 事件 中 )， 可 以 修改 页 面 主题 ， 但 是 无 法 读 取 控 件 
状态 ; 在 页 面 生命 周期 的 后 期 阶段 (如 Button 的 Click 事件 中 )， 可 以 读 到 控件 状态 , 但 是 
却 无 法 修改 页 面 主题 。 要 想 解 决 这 个 矛盾 ， 需 要 一 定 的 编程 技巧 。 下 面 通过 一 个 例子 具体 
说 明 。 

【 例 1-1S】 动态 修改 主题 。 

本 例 演示 如 何以 编程 方式 动态 改变 页 面 主题 。 

(1) 创建 一 个 ASPNET 应 用 程序 。 

(2) 在 项 目 添 加 一 个 主题 ， 分 别 命名 为 GrassGreen 和 SkyBlue。 

(3) 在 GrassGreen 主题 下 添加 一 个 外 观 文件 ， 此 文件 定义 了 DropDownList 和 Button 
的 默认 外 观 ， 代 码 如 下 : 


<asp:Button runat="server" BackColor="lightgreen" BorderStyle="Solid" 
BorderColor="DarkGreen"/> 

<asp:DropDownList runat="server" BackColor="#aaffcc" BorderStyle="Solid" 
BorderColor="Blue"/> 


(4) 在 SkyBlue 主题 下 添加 一 个 外 观 文件 ， 此 文件 也 定义 了 DropDownList 和 Button 
两 种 控件 的 默认 外 观 ， 代 码 如 下 : 

<asp:Button runat="server" BackColor="AliceBlue" BorderStyle="Solid" 

BorderColor="Blue" /> 


<asp:DropDownList runat="server" BackColor="AliceBlue" BorderStyle= 
"Solid" BorderColor="Blue"/> 


(5) 在 项 目 中 添加 一 个 ASPNET 页 面 Pagel.aspx， 页 面 上 放置 一 个 DropDownList 控 
列 出 可 供 选择 的 主题 ， 还 有 一 个 Button 控件 ， 用 以 改变 主题 。 代 码 如 下 : 

请 选择 一 个 主题 <asp:DropDownList ID="DropDownList1l" runat="server"> 
<asp:ListItem Value="GrassGreen"> 草 绿色 </asp:ListItem> 

<asp:ListItem Value="SkyBlue"> 天 蓝 色 </asp:ListItem> 

</asp:DropDownList> 

<asp:Button ID="Buttonl" runat="server" Text=" 更 改 主题 " onclick="Buttonl 
Click” /> 

(6) 为 “更 改 主题 ”按钮 的 Click 事件 编写 代码 ， 以 修改 页 面 主题 。 代 码 如 下 : 


protected void Button1l Click(object sender, EventArgs e) 
| 


件 


this .Theme = DropDownList].SelectedValue; 
(7) 运行 Pagel.asxp 页 面 ， 选 择 一 个 主题 ， 并 单 击 按钮 修改 主题 。 可 以 看 到 程序 运行 
的 结果 并 不 像 所 期 望 的 那样 修改 了 页 面 的 主题 ， 而 是 出 现 以 下 错误 提示 : 
"Theme" 属 性 只 能 在 "Page_PreInit" 事 件 之 中 或 之 前 设置 。 


这 个 错误 提示 证 明了 前 面 关 于 页 面 生命 周期 和 主题 设置 的 讨论 ， 在 服务 器 控件 事件 处 
理 程序 中 不 能 修改 页 面 主题 ， 这 种 思路 行 不 通 。 根 据 提示 信息 ， 很 自然 想到 可 以 尝试 在 页 
面 的 PreInit 事件 中 修改 主题 ， 下 面 来 验证 这 种 思路 。 

(8) 为 页 面 的 PreInit 添加 事件 处 理 程序 ， 并 在 该 事件 处 理 程序 中 设置 页 面 主题 。 代 码 
如 下 : 
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protected void Page Load (object sender, EventArgs e) 


{ 


this.PreInit += new EventHandler (Page PreInit); 


} 


void Page PreInit (object sender, EventArgs e) 


{ 


this .Theme = string s=DropDownList]1.SelectedValue; 


(9) 运行 Pagel.aspx 页 面 ， 选 择 一 个 主题 并 单 击 


看 到 ， 这 个 程序 不 能 改变 页 面 的 主题 


。 这 是 由 于 在 页 面 的 PreInit 


载 完 毕 ， 不 知道 DropDownList 控件 选中 了 哪 一 项 。 


前 面 尝试 的 两 种 方式 都 没有 达到 想 要 的 效果 ， 解 决 这 个 问题 需要 一 种 新 的 思路 。 可 以 
个 QueryString 参数 传递 ， 然 后 在 Page 类 的 PreInit 事件 中 


考虑 把 用 户 所 选择 的 主题 作为 一 
读 取 此 参数 并 修改 主题 。 


(10) 在 Pagel.aspx 页 面 中 添加 一 个 客户 端 按钮 ， 并 为 此 按钮 编写 JavaScript 脚本 。 代 


人 码 如 下 : 


<head runat="server"> 


<title> 动 态 修改 主题 </title> 


<script type="text/javascript" language="javascript"> 


function changeTheme() { 


// 获 得 当前 所 选中 的 主题 


var theme = document. 


ID$S>') .value; 


getElementById('<%=DropDownList1.Client 


// 重 新 加 载 当前 页 面 ， 并 通过 QueryString 传递 所 选中 的 主题 


window.location = 
. 
</ScrEiDE> 
</head> 
<body> 


<input type="button" 
Theme () " /> 


(11) 修改 Page 类 PreInit 事件 的 代码 ， 从 QueryString 中 读 取 新 的 主题 。 代 码 如 下 : 


"pagel .aspx?theme=" + theme; 


value=" 更 改 主题 (客户 端 button) " onclick="change 


void Page_PreInit (object sender, EventArgs e) 


this .Theme = Request.QueryString["theme"]??"GrassGreen"; 


(12) 运 生 


动态 修改 主题 
文件 人 蝙 辑 下 ) 查看 Q) 


历史 G) 书签 @) 工具 C) 帮助 00 


了 此 页 面 ， 并 修改 主题 ， 程 序 运 


a Firefoxr 


行 正常 。 运 行 界面 如 图 1.18 所 示 。 
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1.4 Web 服务 


Web 服务 (Web Services) 是 一 种 面向 服务 的 架构 技术 ， 通 过 标准 的 Web 协议 提供 服 
务 ， 目 的 是 保证 不 同 平台 的 应 用 服务 可 以 互 操作 。 利 用 Web 服务 ， 可 以 实现 跨 操作 系统 、 
跨 应 用 程序 、 跨 编程 语言 的 应 用 程序 互 操 作 。 例 如 ， 一 个 运行 在 Linux 操作 系统 上 的 Java 
程序 可 以 调用 一 个 运行 在 Windows 操作 系统 上 的 用 .NET 开发 的 Web 服务 ， 反 之 亦 然 。 


1.4.1 Web 服务 简介 


Web 服务 实现 了 在 异类 系统 之 间 以 XML 消息 的 形式 进行 数据 交换 。 在 Web 服务 出 现 
以 前 ， 人们 也 曾 尝试 过 几 种 在 不 同 应 用 程序 之 问 通 信 的 方式 , 如 DCOM、 IIOP 和 Java/RMI 
等 ， 这 些 方式 要 求 在 客户 端 和 服务 器 之 间 进行 紧密 集成 ， 并 使 用 特定 的 二 进 制 数 据 格式 。 
相对 于 以 前 出 现 的 这 些 技术 来 说 ，Web 服务 对 通信 双方 的 要 求 更 加 宽松 ， 只 要 求 接收 方 可 
以 理解 收 到 的 消息 。 

Web 服务 的 基础 是 HTTP 和 XML。Web 服务 使 用 HTTP 协议 进行 网 络 通信 ， 用 XML 
描述 通信 过 程 中 传输 的 数据 。 除 HTTP 和 XML 这 两 种 最 基础 也 最 常见 的 技术 外 ，Web 服 
务 还 要 用 到 以 下 3 个 协议 : 

口 简单 对 象 访问 协议 SOAP (Simple Object Access Protocol): 是 一 种 标准 化 的 通信 规 

范 ， 主 要 用 于 Web 服务 (web service) 中 。SOAP 的 出 现 是 为 了 使 Web 服务 器 在 
检索 XML 数据 时 无 须 花 时 间 去 格式 化 页 面 ， 并 能 够 让 不 同 应 用 程序 之 间 通 过 
HTTP 通信 协定 ， 以 XML 格式 互相 交换 彼此 的 数据 ， 使 其 与 编程 语言 、 平 台 和 硬 
件 无 关 。 

口 Web 服务 描述 语言 WSDL (Web Services Description Language) : WSDL 描述 Web 
服务 的 公共 接口 。 这 是 一 个 基于 XML 的 关于 如 何 与 Web 服务 通信 和 使 用 的 服务 
描述 ， 也 就 是 描述 与 目录 中 列 出 的 Web 服务 进行 交互 时 需要 绑 定 的 协议 和 信息 
格式 。 

口 统一 描述 、 发 现 和 集成 UDDI (Universal Description, Discovery, and Integration ) : 
是 一 个 基于 XML 的 跨 平 台 的 描述 规范 , 可 以 使 世界 范围 内 的 企业 在 互联 网 上 发 布 
自己 所 提供 的 服务 。 


1.4.2 创建 Web 服务 


1.4.1 节 讲 解 的 知识 来 看 ，Web 服务 涉及 多 个 协议 的 解析 处 理 ， 是 一 个 很 复杂 的 工 
作 。ASPNET 对 这 些 协议 进行 了 封装 ， 使 开发 人 员 可 以 将 注意 力 集中 于 服务 本 身 而 非 通信 
协议 。 当 使 用 ASPNET 进行 基本 的 Web 服务 开发 时 ， 并 不 要 求 开 发 人 员 理 解 这 些 底层 协 
议 ， 如 SOAP、WSDL 等 ， 开 发 人 员 可 以 方便 快速 地 开发 基于 Web 服务 的 应 用 程序 。 本 节 
和 14.1.3 节 将 通过 具体 例子 来 说 明 如 何 创 建 和 访问 Web 服务 。 
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【 例 1-16】 创建 Web Service。 
本 例 演 示 如 何 创 建 一 个 Web Service。 本 例 创建 的 Web Service 只 有 一 个 功能 : 将 公历 


日 期 转换 为 中 国 农历 日 期 。 


(1) 在 Visual Studio 开发 环境 中 ， 从 菜单 栏 中 选择 “文件 ”| “新建 ”|“ 项 目 ” 命 令 ， 


在 弹出 的 “新 建 项 目 ” 对 话 框 中 ， 选 择 “ASPNET Web 服务 应 用 程序 ” 单 击 “确定 ” 按 


钮 。 


项 目 创建 后 ， 会 生动 生成 一 个 Servicel.asmx 文件 ， 这 就 是 Web 服务 文件 ，asmx 是 


ASPNET Web 服务 的 文件 扩展 名 。 


(2) 打开 Servicel.asmx.cs 文件 ， 可 以 看 到 如 下 代码 : 


[WebService (Namespace = "http://tempuri.org/")] 
[WebServiceBinding (ConformsTo = WsiProfiles.BasicProfilel 1)] 
[System.ComponentModel .ToolboxItem(false)] 
public class Servicel : System.Web.Services.WebService 
{ 

[WebMethod] 

public string HelloWorld() 

{ 

return "Hello World"; 

} 

} 


上 述 代码 中 ,在 HelloWorld 方法 前 面 的 [WebMethod] 表 示 此 方法 是 一 个 Web 服务 方法 ， 


客户 端 可 以 通过 Web 服务 方式 调用 此 方法 。 自 动 生成 的 class Servicel 前 面 的 代码 现在 暂 
时 不 需要 理解 。 


(3) 修改 Servicel.asxm.cs 的 代码 。 删 除 自动 生成 的 HelloWorld 方法 ， 添 加 一 个 


GetLunarDate() 方 法 。 注 意 要 在 GetLunarDate 前 面 添 加 WebMethod 标志 。 代 码 如 下 : 


// 得 到 农历 日 期 的 Web 服务 

[WebService (Namespace = "http://tempuri.org/")] 
[WebServiceBinding (ConformsTo = WsiProfiles.BasicProfilel 1)] 
[System.ComponentMode1l .ToolboxItem(false)] 

public class Servicel : System.Web.Services.WebService 


{ 
// 汉 字数 字数 组 
string[] numbers = 
(i 
HO hs HH 
th = eh eo te 
/// Web 服务 方法 ， 以 字符 串 形式 返回 公历 日 期 对 应 的 农历 日 期 
/// <param name="date"> 要 转换 的 公历 日 期 </param> 
/// <returns> 转 换 后 的 农历 日 期 </returns> 
[WebMethod] 
public string GetLunarDate (DateTime date) 
System.Globalization.ChineseLunisolarCalendar calendar = new 
System.Globalization.ChineseLunisolarCalendar (); 
string result =""; 
result = result + toChineseNum(calendar.GetYear()) + "年 "; 
// 得 到 农历 年 
result = result + numbers[calendar.GetMonth (date)]+" 有 月 "; 
// 得 到 农历 月 
int month=calendar .GetDayOfMonth (date); // 得 到 农历 日 
// 如 果农 历 日 小 于 10， 则 按照 习惯 ， 应 该 在 对 应 汉字 前 加 一 个 “ 初 ” 字 
// 如 果农 历 日 大 于 10， 则 不 加 “ 初 ” 字 
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if (month < 11) 
result = result + " 初 " + numbers [month]; 
else 
result = result + numbers[month]; 
return result; 
} 
// 将 整数 转换 为 汉字 数字 形式 (如 2010 转换 为 二 零 一 零 ) 
private string toChineseNum(int n) 
1 
A 
string result = ""; 


while (n > 0) // 此 循环 对 每 位 数字 进行 转换 


return result; 


} 

(4) 在 Servicel.asmx 文件 上 右 击 ,从 弹出 的 快捷 菜单 中 选择 “在 浏览 器 中 查看 ”选项 ， 
则 可 以 在 浏览 器 中 打开 这 个 Web 服务 ， 页 面 中 显示 了 对 于 此 服务 的 简单 说 明 ， 如 图 1.19 
所 示 。 


Servicel Yeb 服务 - Worille Firefox 


文件 @) 编辑 到) 查看 W) 历史 E) 书签 ) 工具 CD) 下 助 Q0) 

WE ec x "Cr 
eos 本 媚 回 - 和 -M+- 安 - 口 - 当 a-:@ 

| | Servicel Web 服务 


支 乔 下 列 雹 作 。 有 关 正式 定义 ， 请 查看 潜 务 及 明 . 


® GetLunarDate 


此 Web 服务 使 用 http://tempuri.org/ 作为 默认 命名 空间 
建议 : 公开 XML Web services 之 前 ， 请 更 改 默认 命名 空间 . 


每 个 XML Web services 部 需要 一 个 唯一 的 命 交 间 ， 以 便 客 户 端 应 用 程序 能 够 插 它 与 Web 上 的 其 他 忠 务 区 分 开 。 http;//tempuri,org/ 可 用 于 处 
于 开发 阶段 的 XML Web services， 而 已 发 布 的 XML Web services 应 使 用 更 为 水 久 的 命名 交 间 - 


应 使 用 货 按 制 的 命 考 奖 间 来 标识 XML Web services- 例 和 如， 可 以 使 用 公司 的 Internet 域 洗 作为 命名 奖 间 的 一 部 分 。 尽 千 有 许多 XML Web 
» 


区 


EE 
图 1.19 Servicel Web 服务 页 面 


从 图 1.19 所 示 的 Servicel.asmx 页 面 中 可 以 看 到 ，Servicel 包含 一 个 GetLunarDate() 方 
法 ， 这 就 是 在 代码 中 标识 为 WebMethod0 的 方法 。 

(5) 在 图 1.19 所 示 的 Servicel.asmx 页 面 中 ， 单 击 页 面 顶部 的 “服务 说 明 ” 超 链接 ， 
可 以 转 到 此 服务 对 应 的 WSDL 页 面 ， 如 图 1.20 所 示 。 

从 图 1.20 可 以 看 到 ，ASPNET 自动 生成 了 Web Service 的 WSDL， 从 而 把 开发 人 员 从 
底层 协议 中 解放 出 来 。 

(6) 在 图 1.19 所 示 的 Servicel.asmx 页 面 中 ， 单 击 GetLunarDate() 方 法 链接 ， 会 转 到 此 


= 
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方法 的 说 明 页 面 ， 如 图 1.21 所 示 。 


—<wsdl:definitions targetNamespace= “http://tempuri. org/ ”> 
一 人 sdl :types> 
-<s:schema elementFormDefault=“qualified”targetNamespace= “http://tempuri. org/ > 
-<s:element name=“GetLunarDate “> 
-<s:complexType> 
-<s:sequence> 
《s:element minOccurs="1” maxOccurs="1” name="date” type="s:dateTime”/> 
</s:sequence> 
</s:complexType> 
</s:element> 
-<s:element name="GetLunarDateResponse”> 
-<s:complexType> 


1.20 Servicel 的 WSDL 页 面 


1 Yeb 服务 - orille Firefox lx 


Oo 一 


| 
By | | http://1ocalhost:2212/Servicel. asmx?op=GetLunarDate 全 > Wlea: A 
-和 M: 安 a 


GetLunarDate 
测试 


若 要 使 用 HTTP POST 协议 对 怒 作 进行 测试 ， 请 单 击 * 油 用 “ 按 钥 - 
参数 位 


date; 
SOAP 1.1 
以 下 是 SOAP 1,2 请 求 和 响应 示例 。 所 显示 的 占 位 特需 梦 换 为 实际 伞 - 


图 1.21 GetLunarDate0 方 法 说 明 页 面 


在 图 1.21 所 示 页 面 中 ， 显 示 了 GetLunarDate() 方 法 所 需 的 参数 ， 还 显示 了 此 方法 对 应 
的 SOAP 消息 示例 。 在 此 页 面 上 ， 给 GetLunarDate() 方 法 输入 参数 ， 单 击 “ 调 用 ”按钮 ， 
即 可 通过 Web 服务 调用 此 方法 ， 并 在 浏览 器 中 显示 返回 结果 ， 如 图 1.22 所 示 。 

(7) 如 果 要 让 Web Service 脱离 开发 环境 运行 ， 则 需要 将 其 发 布 。 发 布 Web Service 的 
方法 与 发 布 Web 应 用 程序 相同 。 在 Visual Studio 解决 方案 资源 管理 器 中 , 右 击 Web Service 
项 目 名 称 ( 本 例 为 WebService1)， 从 弹出 的 快捷 菜单 中 选择 “发 布 选 项, 则 弹出 如 图 1.23 
所 示 的 “发 布 Web” 对 话 框 。 从 “发 布 方法 ”下 拉 列 表 框 中 选择 “文件 系统 ” 在 “目标 
位 置 ”文本 框 中 设置 发 布 路 径 后 ， 单 击 “ 发 布 ” 按 钮 ， 即 可 完成 发 布 。 
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发 布 使 用 “项 目 属性 ”的 “打包 /发 布 Yeb” 和 “打包 /发 布 S9L” 选项 卡 中 的 设置 , 


Mozilla Firefox 


文件 中” 编辑) 查看 W) 历史 GE) 书签 @) | 工具 中 下 助 吕 
[Oe Ce - lea 
Coe 5q 6@ 


上 http: [lecalhoes---zfGetLanarate| 二 - 发 和 
生成 配置 -Debug 
扇 帅 生成 民 寻 所 要 状 们 更 必 配 豆 
1 文件 并 未 包含 任何 关联 的 样式 信息 。 文 档 树 显示 如 ~ ga 可 
目标 位置 :和 [区 
《Latringy 二 夫 地 作 年 十 一 月 初 /stringy Re 
[ 语 六 


图 1.22 GetLunarDate() 方 法 调用 结果 图 1.23 发 布 Web 服务 


(8) 在 Visual Studio 中 完成 Web Service 的 发 布 后 ， 会 把 Web Service 相关 文件 都 复制 
到 用 户 所 指定 的 目录 下 。 接 下 来 ， 还 需要 将 其 部 署 到 目标 服务 器 上 。 主 要 操作 步骤 是 ， 找 
到 发 布 的 目标 文件 夹 ， 把 文件 夹 复制 到 Web 服务 器 上 ， 并 将 此 文件 夹 设 置 为 Web 共享 ， 
设置 相应 的 访问 权限 (如 是 否 允许 匿名 访问 、 列 出 目录 、 读 写 权 限 等 )。 


外 提示 : 如 何 判断 一 个 ASP.NET Web 服务 是 否 部 署 成 功 ? 方法 是 在 浏览 器 中 输入 Web 服 
务 地 址 ， 如 果 出 现 了 Web 服务 描述 页 面 (如 图 1.19 所 示 ) ， 则 Web 服务 部 署 
成 功 。 


1.4.3 访问 Web 服务 


1.4.2 节 介绍 了 创建 Web 服务 的 过 程 。Web 服务 创建 后 ， 公 开 了 一 系列 操作 接口 ， 各 
种 客户 端 应 用 程序 (包含 Web 应 用 、PC 上 的 WinForm 应 用 、 移 动 设备 应 用 程序 ) 都 可 以 
按照 Web 服务 标准 访问 此 接口 ， 享 受 Web 服务 所 提供 的 功能 。 

如 前 所 述 , 由 于 ASPNET 对 Web 服务 底层 协议 进行 了 很 好 的 封装 , 开发 人 员 开发 Web 
服务 应 用 程序 变 得 简单 高 效 。 从 ASPNET 中 访问 一 个 Web 服务 也 很 容易 实现 。 下 面 通过 
一 个 例子 说 明 如 何 访问 Web 服务 。 

【 例 1-17】 访问 Web 服务 。 

本 例 演示 如 何 访问 例 1-16 所 创建 的 公历 日 期 转 农历 日 期 的 Web 服务 。 

(1) 创建 一 个 ASPNET Web 应 用 程序 ， 命 名 为 WebServiceClient。 

(2) 运 行 例 1-16 所 创建 的 Web 服务 。 运 行 这 个 服务 有 两 种 方式 , 一 种 是 从 Visual Studio 
开发 环境 中 运行 ， 一 种 是 将 其 发 布 和 部 署 到 Web 服务 器 再 运行 。 这 两 种 方式 都 可 以 , 但 是 
一 定 要 保证 Web 服务 正确 运行 。 检 测 Web 服务 是 否 正常 运行 的 方法 是 在 浏览 器 中 查看 此 
服务 ， 如 果 服 务 正 常 运行 了 ， 把 服务 的 URL 地 址 复制 下 来 ， 在 下 一 步骤 中 将 用 到 此 地 址 。 

(3) 在 WebServiceClient 项 目 中 右 击 ,在 弹出 快捷 菜单 中 选择 “添加 Web 引用 ”命令 ， 
则 弹出 “添加 Web 引用 ”对 话 框 。 在 对 话 框 的 地 址 栏 中 输入 前 一 步 所 获得 的 Web 服务 地 
址 ， 则 对 话 框 中 会 显示 与 此 Web 服务 相关 的 描述 信息 ， 如 图 1.24 所 示 。 
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EEC 3 
请 定位 到 提供 Web 服务 的 URL， 然后 单 击 “ 添 加 引用 ”， 添加 位 于 该 VRL 上 的 所 有 可 用 服务 。 

(7 ee 

击 fen:/71ochost:22127Servicel omz  。” 匡 | 回 衣 和 


位 于 此 WRL 上 的 Web 服务 G): 
1 - 


支持 下 列 操作 。 有 关 正式 定义 ， 请 查看 服务 说 明 . 


。 GetLunarDate 


Yeb 引用 名 QD; 


此 Web 服务 使 用 http://tempuri.org/ 作为 默认 命名 空间 
hocalhost 


建议 : 公开 XML Web services 之 前 ,请 更 改 默认 命名 空间 


[还 ms | 
每 个 XML Web services 都 需要 一 个 唯一 的 命名 空间 ,以便 客 户 端 应 用 三 
取消 
A 


程序 能 够 格 它 与 Web 上 的 其 地 服务 区 分 开 ，http://tempuri.org/ 可 用 
于 处 于 开发 阶段 的 XML Web services， 而 已 发 布 的 XML Web 
services 应 使 用 更 为 永久 的 命名 空间 。 


应 使 用 您 控制 的 命名 空间 来 标识 XML Web services。 例 如 ,可 以 使 用 
公司 的 Internet 域名 作为 命名 空间 的 一 部 分 。 尽 管 有 许多 XML Web 
指向 WPh 上 的 守 际 前 源 - 本 


Seryires 命名 空间 看 们 _URIL， 但 它 f 


图 1.24 添加 Web 引用 
(4) 在 图 1.24 所 示 对 话 框 中 单 击 “添加 引用 ”按钮 ， 则 Visual Studio 会 自动 生成 一 组 
类 ,这 些 类 封装 了 调用 指定 的 Web 服务 所 需要 的 信息 。 可 以 通过 对 象 浏览 器 窗口 查看 自动 
生成 的 类 的 相关 信息 ， 如 图 1.25 所 示 。 


目 @) 生成 @) 调式 W) 国 队友 ) 数据 W) 工具 CD) 体系 结构 C) 测试 GG) 分 析 加 ) 


文件 四 
更 助 0 

DBdG% NI. mB Dr || 殉 [= Ib er kr > 
发 大: | 创建 发 布设 置 区 


编辑 下) 窗口 人 0 


5% 当 外 


EEE 


< 搜索 > 
Systenm. Web. DynanicData 二 CancelAsyne (object) 
四- 口 Systen Web, Entity 9 
Systen. Web. Extensions GetLunarDateAsync (Systemn .Datel 
Systen Web. Extensions. Desien 让 GetLunarDateAsyne Systen, Date 
图 - 口 Systen Web Hobile IsLecalfilssystemfetservice (string) 
System. Web. Services SY OnGetLunarDateDperationConpleted(object) 


器 本 号 喜 潍 名 


bb 
二 
3 


Systen Xnl YServicel 0 
Systen. Xnl. Ling 河上 
TheneSanple 河 wsepefenltcredentials 
由 - 团 sercontrolSanpls a GetLanarDlate0perationConpleted 
由 -4 园 WebServicel 2 useDefaultCredentialsSetExplicitly 
日 天 Yebserviceclient 多 GetLanarDateCompleted 
EW-{} WebServiceClient 
El- {) YebServiceClient. localhost 
tLunarDateConpletedEventAres 
固 国 GetLunarDateCompletedEventHandler 
9 ,i 


Ee El 苹 国 愉 天 各 上 


号 说 识 列表 国 Wz 结果 电 六 
就 绪 


图 1.25 添加 Web 引用 后 自动 生成 的 类 


图 1.25 中 ， 自 动 生成 的 这 些 类 的 名 字 与 创建 Web Service 时 使 用 的 类 名 相同 ， 但 却 是 
不 同 的 类 。 在 客户 端 自动 生成 的 类 其 实 是 服务 器 端 相应 类 的 代理 ,而 不 是 服务 器 端 类 本 身 。 
客户 端 代理 类 模拟 了 服务 器 端 相 应 类 的 接口 ， 从 而 允许 用 户 像 使 用 服务 器 端 类 那样 来 使 用 
代理 类 。 另 外 ， 代 理 类 在 Web Service 中 起 着 通信 桥梁 的 作用 ， 客 户 通过 代理 类 向 Web 服 
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务 发 出 请 求 ，Web 通过 代理 向 客户 返回 结果 。 


各 注意 : 添加 Web 引用 后 自动 生成 的 类 与 Web 服务 中 的 类 虽然 名 字 相 同 ， 但 却 是 不 同 的 
类 ， 前 者 是 后 者 的 代理 ， 要 注意 区 分 这 两 者 。 


(5) 在 WebServiceClient 项 目 中 添加 一 个 页 面 Defaultaspx， 页 面 上 放置 一 个 Calendar 
和 一 个 Label， 页 面 代 码 如 下 : 


<form id="forml" runat="server"> 
<div> 选 择 一 个 阳历 日 期 查看 其 对 应 的 农历 日 期 ; 
<asp:Calendar ID="Calendarl" runat="server" 
onselectionchanged="Calendqar1l SelectionChanged"></asp:Calendar> 
br /> 
<asp:Label ID="Labell" runat="server" Text="Label"></asp:Label> 
</div> 
</form> 


(6) 当 用 户 从 Calendar 控件 中 选择 一 个 日 期 时 , 后 台 程 序 会 调用 Web 服务 把 此 日 期 转 
换 为 农历 并 显示 在 页 面 上 。 为 此 ， 需 要 在 Calendar 控件 的 SelectionChanged 事件 中 编写 如 
下 代码 。 


protected void Calendarl SelectionChanged (object sender, EventArgs e) 
{ 
/ /创建 一 个 Web Service 代理 类 的 实例 
localhost.Servicel service = new WebServiceClient.localhost. 
Servicel (); 
// 调 用 Web Service， 得 到 农历 日 期 
string lunar=service.GetLunarDate (Calendarl]l .SelectedDate); 
// 把 阳历 和 农历 日 期 都 显示 在 Label 上 
Label1.Text = string.Format ("{0:D} 所 对 应 的 农历 是 {1}"，Calendarl .Selected 
Date, lunar); 
service.Dispose(); // 释 放 代理 资源 


(7) 运行 此 页 面 ， 从 Calendar 控件 中 选择 一 个 日 期 , 下方 就 会 显示 出 对 应 的 农历 日 期 ， 
如 图 1.26 所 示 。 


文件 全 ) 编辑 在) 查看 WW | 历史 G) 书签 @) 工具 G) 帮助 只 
Ce x [lms | 
oe 


图 1.26 Web 服务 客户 端 运行 界面 
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1.4.4 Web Service 实例 一 一 生活 小 助手 


软件 的 集成 和 互 操 作 是 软件 发 展 的 一 个 重要 方向 ，Web 服务 正 是 顺应 了 这 一 方向 ， 提 
供 了 跨 平台 的 软件 互 操作 手段 。Web 服务 的 应 用 发 展 迅速 ， 当 前 的 Intemet 有 许多 公司 提 
供 各 种 Web 服务 ， 有 的 服务 收费 ， 也 有 免费 服务 ， 这 些 服务 内 容 包 含 中 英文 互 译 、 天 气 预 
报 、 股 市 行情 查询 、 邮 政 编码 查询 、 电 话 区 号 查询 、 飞 机 航班 、 列 车 时 刻 查 询 、IP 地 址 查 
询 等 。 基 于 Intemet 存在 的 众多 Web 服务 ， 程 序 员 可 以 轻松 开发 出 功能 强大 的 应 用 程序 。 

【 例 1-18】 生活 小 助手 。 

本 例 利用 当前 Intemet 上 存在 的 一 些 免费 Web 服务 ， 开 发 了 一 个 小 实用 程序 ， 该 程序 

可 以 实现 中 英文 互 译 和 天 气 预 报 功能 。 这 两 个 功能 都 是 人 们 在 工作 和 生活 中 经 常用 到 的 ， 
所 以 把 这 个 程序 称 为 生活 小 助手 。 
于 本 例 依赖 于 Intemet 上 现存 的 Web 服务 , 如果 被 依赖 的 Web 服务 发 生 更 改 或 者 关 
闭 ， 那 么 本 例 将 不 能 正常 运行 。 如 果 读 者 在 阅读 本 书 时 ， 本 例 所 引用 的 Web 服务 已 经 不 能 
使 用 ,读者 可 以 从 网 上 搜索 其 他 类 似 的 Web 服务 ， 按 照 本 例 所 讲 的 方法 ， 参 照例 子 中 的 代 
码 ， 实 现 类 似 的 功能 。 

(1) 创建 一 个 ASPNET Web 应 用 程序 ， 命 名 为 Assistant。 

(2) 生活 小 助手 程序 包含 两 个 功能 ， 首 先 实现 英汉 互 译 功能 。 在 项 目 中 添加 英汉 互 译 
的 Web 服务 引用 ， 地 址 为 http://fy.webxml.com.cn/webservices/EnglishChinese.asmx。 这 个 
Web 服务 包含 多 个 方法 , 本 例 主 要 用 到 其 中 的 一 个 TranslatorString() 方 法 ， 此 方法 接收 一 个 
string 类 型 参数 ， 表 示 需 要 翻译 的 词语 (中 英文 都 可 以 )， 返 回 一 个 字符 串 数组 ， 表 示 翻 译 

(3) 在 项 目 中 添加 一 个 Default.aspx 页 面 。 

(4) 这 个 程序 包含 两 个 功能 ， 每 个 功能 在 页 面 上 表现 为 一 个 div， 为 了 实现 统一 布局 ， 
在 页 面 中 定义 一 个 CSS， 代 码 如 下 : 


div.assistant 


{ 


border:dashed lpx silver; /* 边 框 银灰 色 细 虚线 */ 
width:300px;height:150px; /* 宽 300， 高 150*/ 
background-color:#ccddff; /* 背 景 浅 蓝 色 */ 
margin:10px; /* 边 距 为 10 个 像素 */ 
float:left; /* 向 左 浮动 */ 


(5) 在 Defaultaspx 页 面 中 添加 一 个 div 以 及 相关 控件 ， 实 现 翻译 界面 。 代 码 如 下 : 


<div class="assistant"> 

<h3> 英 汉 双 向 翻译 </h3> 

输入 单词 : <asp:TextBox runat="server" ID="word" /> 
<asp:Button ID="Buttonl" runat="server" Text=" 翻 译 " onclick="Buttonl 
Click” /><br /> 
<asp:Label ID="translation" runat="server" ></asp:Label> 

</div> 


(6) 为 “翻译 ”按钮 编写 代码 ， 调 用 Web Service 实现 翻译 功能 并 显示 结果 ， 代 码 
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如 下 : 


protected void Button1l Click(object sender, EventArgs e) 
T 
webservice.EnglishChinese utility = new 
BusinessAssistant .webservice.EnglishChinese(); 
// 生 成 Web 服务 代理 
try 
{ 
string[] result = utility.TranslatorString (word.Text); 
// 调 用 Web 服务 进行 翻译 
// 把 翻译 结果 显示 在 页 面 上 
string temp = ""; 
temp += "读音 : " + result[1] + "<br/>"; 
temp += "翻译 : " + result[3] + "<br/>"; 
translation.Text = temp; 


} 

catch 

; translation.Text = "翻译 过 程 中 出 现 错误 。"; // 如 果 出 现 异常 ， 则 显示 提示 信息 
a 

utility.Dispose(); 


(7) 运行 Default.aspx 页 面 ， 运 行 结果 如 图 1.27 所 示 。 
(8) 前 面 几 个 步骤 实现 了 英汉 互 译 功能 ， 下 面 再 来 实现 天 气 预报 功能 。 本 例 所 选用 的 


天 气 预报 Web 服务 地 址 为 http://www.ayandy.com/Service.asmx， 在 项 目 中 添加 对 这 个 Web 
服务 的 引用 。 这 个 Web 服务 包含 多 个 方法 ， 本 例 只 用 到 其 中 一 个 getWeatherByCityName() 
方法 。 此 方法 接收 两 个 参数 : 第 一 个 参数 为 string 类 型 ， 表 示 要 查询 的 城市 名 称 ， 第 二 个 
参数 为 枚 举 类 型 ， 表 示 要 查询 哪 一 天 的 天 气 〈 可 以 是 今天 、 明 天 或 者 后 天 )。 此 方法 返回 一 
个 字符 串 数 组 ， 表 示 查 询 到 的 结果 


(9) 在 Default.aspx 页 面 添加 另 一 个 div， 并 在 其 中 放置 一 个 TextBox、 一 个 Label 和 


一 个 DropDownList， 作 为 天 气 预报 查询 界面 。 代 码 如 下 : 


<div class="assistant"> 
<h3> 天 气 预报 查询 </h3> 
要 查询 的 城市 : <asp:TextBox runat="server" ID="city" /><br /> 
要 查询 的 日 期 : <asp:DropDownList ID="date" 
runat="server"> 
<asp:ListItem Value="0"> 今 天 </asp:ListItem> 
<asp:ListItem Value="1"> 明 天 </asp:ListItem> 
<asp:ListItem Value="2"> 后 天 </asp:ListItem> 
</asp:DropDownList> 
<asp:Button ID="Button2" runat="server" 
Text=" 查 询 " onclick="Button2 Click" /><br /> 
<asp:Label ID="weather" runat="server" ></asp:Label> 
</div> 


(10) 为 “查询 ”按钮 编写 代码 ， 实 现 天 气 预报 查询 和 显示 ， 代 码 如 下 : 


protected void Button2 Click(object sender, EventArgs e) 
i 
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1 


// 生 成 Web 服务 代理 


webservicel .Service service = new BusinessAssistant.webservicel. 
Service(); 
int day = int.Parse(date.SelectedValue); // 得 到 用 户 所 选择 的 日 期 
Et 
{ 
// 调 用 Web 服务 获得 天 气 预报 
string[] result = service.getWeatherbyCityName (city.Text, 
(webservicel .theDayFlagEnum) day); 
string temp; 
// 得 到 城市 名 称 和 日 期 
temp = string.Format ("<br/>{0}{1} 的 天 气 情况 <br/>", result[1],result 
5]); 
人 4= EsULtl2] tT esult ll" "Troultlalt "> <br/> 
// 得 到 天 气 、 温 度 和 风力 
temp += "<img src="'" + result[6] + "' alt=' 天 气 图 片 '/>"; 


// 得 到 天 气 图 片 
weather.Text = temp; // 把 以 上 信 息 显示 在 页 面 上 
a // 异 常 处 理 
; weather.Text = "查询 过 程 中 出 现 错误 。"; 
Cpa 
: service.Dispose(); // 释 放 资 源 


(11) 运行 Default.aspx 页 面 ， 运 行 结果 如 图 1.28 所 示 。 


文件 中 编 得 加、 查看 WW) 历史 E) 书签 @ 工具 人 ) 


x 
文件 中 闹 强 人 E) 查看 WD 历史 G) 书签 如 工具 DD 帮助 
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生活 小 助手 英汉 双向 鼻 译 天 气 预报 查询 
输入 单词 ，| 计 算 要 查询 的 城市 : HF 比 京 
菊 汉 双向 翻译 te lone to canvate | OPW | EA S| | 
he Fi 
el | 
制 译 n。 神 试 , 试验 ，v。 测 试 , 试验 ,接受 测验 
冶 Co 
[ 碱 展 怕 让 ED EaE 
图 1.27 翻译 助手 运行 界面 图 1.28 生活 小 助手 运行 界面 


1.5 用 户 控 件 


ASPNET 用 户 控件 与 完整 的 ASPNET 页 面相 似 ， 可 以 同时 具有 用 户 界 面 页 和 代码 。 
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处 理 程 序 。 使 用 用 户 控件 可 以 提高 程序 的 模块 化 和 复 用 性 。 与 页 面 所 不 同 的 是 ， 用 户 控件 
不 是 一 个 完整 页 面 ， 不 能 在 浏览 器 中 进行 浏览 。 用 户 控件 文件 的 扩展 名 是 ascx，ascx 文件 
的 第 一 条 语句 是 <%@Control 指令 。 


1.5.1 创建 和 使 用 用 户 控件 


相对 于 ASPNET 自 带 的 系统 控件 来 说 ， 用 户 控 件 是 开发 人 员 自 行 定义 和 创建 的 控件 。 
用 户 控件 继承 自 System.Web.UI.UserControl 类 。 创建 用 户 控 件 的 过 程 与 创建 普通 ASPNET 
页 面 很 相似 。 用 户 控 件 不 能 作为 单独 页 面 运 行 ， 必须 嵌入 到 ASPNET 页 面 中 使 用 。 用 户 控 
件 可 以 包含 其 他 控件 , 包含 ASPNET 标准 控件 和 其 他 用 户 控 件 。 使 用 用 户 控件 时 必须 先 注 
册 再 使 用 。 下 面 通过 一 个 例子 演示 如 何 创建 和 使 用 用 户 控 件 。 

【 例 1-19】 创建 和 使 用 用 户 控 件 。 

许多 Web 应 用 程序 的 页 头 和 页 脚 都 是 固定 的 内 容 ， 页 头 一 般 是 网 站 横幅 和 链接 ， 页 肢 
一 般 是 版 权 信 息 和 联系 信息 。 可 以 把 这 两 部 分 内 容 分 别 做 成 用 户 控 件 ， 提 高 页 面 模块 化 程 
度 和 可 读 性 。 本 例 演 示 如 何 创建 和 使 用 页 头 、 页 脚 用 户 控件 。 本 例 来 源 于 笔者 开发 的 《新 
型 农村 合作 医疗 管理 信息 系统 》 项 目 ， 做 了 少许 改动 。 由 于 本 例 代 码 较 多 ， 首 先 给 出 例子 
完成 后 的 界面 如 图 1.29 所 示 ， 使 读者 有 一 个 整体 概念 ， 然 后 再 逐步 介绍 其 实现 过 程 。 


文件 刀 ”编辑 E) 查看 历史 G) 书签 @) 工具 CD) 帮助 如 


eam 
“EEE 


北京 “名 
天 气 


当前 日 期 ，2010-02-02 Tuesday 


图 1.29 页 头 页 脚 用 户 控件 示例 


(1) 创建 一 个 Web 应 用 程序 ， 命 名 为 UserControlSample。 

(2) 在 “解决 方案 资源 管理 器 ”中 ， 右 击 项 目 名 称 ， 从 弹出 的 快捷 菜单 中 选择 “添加 ” 
|“ 新 建 项 ”命令 ， 在 弹出 窗口 中 选择 “Web 用 户 控件 ”， 重 命名 为 Weather， 然 后 单 击 “ 添 
加 ”按钮 ， 如 图 1.30 所 示 。 

(3) Weather 用 户 控件 的 作用 是 通过 Web 服务 获取 和 显示 天 气 信息 。 控 件 采 用 Table 
布局 ， 把 天 气 文字 信息 和 图 片 信息 整齐 地 显示 在 页 面 上 。 通 过 Web Service 获取 天 气 的 具 
体 实现 原理 在 Web Service 一 节 中 已 经 介绍 ， 这 里 仅 给 出 主要 代码 。Weatherascx 的 关键 代 
码 如 下 : 
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二 了 
日 wua ce 


局 划 类 型 : Visual C# 
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bd 
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区 ASP NET 服务 器 控件 Visudl C# 


图 1.30 添加 Web 用 户 控件 


<table> 

tx 

<td><asp:Label ID="cityName" runat="server" Text=" 城 市 名 称 "></asp:Label><br 
VSR Ed 

<td><asp:Image ID="weatherImage" runat="server" Height="32px" Alternate 
Text=" 天 气 " /></td> 

<td> 

<asp:Label ID="weatherDesc" runat="server" Text=" 天 气 情况 "></asp:Label> 
</td> 

</tr> 

</table> 


Weather 用 户 控件 后 台 代码 文件 Weatherascx.cs 代码 如 下 : 


protected void Page Load(object sender, EventArgs e) 
{ 
ShowWeather (); 
} 
// 在 页 面 上 显示 天 气 信息 
Private void showWeather () 
{ 
try 
{ 
Var service = new weather.Service(); // 创 建 Web 服务 代理 
// 得 到 某 城市 (此 处 为 北京 》 当 天 天 气 信息 
string [] result=service.getWeatherbyCityName ("北京 "，weather. 
theDayFlagEnum.Today); 
service.Dispose(); / /释放 资源 
// 在 页 面 上 显示 天 气 描述 信息 和 图 片 信息 
weatherDesc.Text=result [2]+"<br/>"+result [3]; 
weatherImage.ImageUrl=result [6]; 
上 
Catch // 发 生 异 常 
{ 
weatherDesc.Text = "未 能 获取 天 气 信息 "; 
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(4) 在 项 目 中 添加 一 个 新 的 用 户 控件 Headerascx， 用 于 显示 页 面 头 部 信息 。Header 用 
户 控件 包含 了 Weather 用 户 控件 ， 下 面 介 绍 如 何 把 Weather 用 户 控件 添加 到 Header 用 户 控 
件 中 。 打 开 Headerascx 页 面 并 切换 到 设计 视图 ， 然 后 从 “解决 方案 资源 管理 器 ”中 ， 把 
Weather.ascx 用 户 控件 拖 动 到 Headerascx 的 设计 视图 , 从 而 完成 把 Weather 添加 到 Header。 
此 时 打开 Headerascx 文件 ， 可 以 发 现 有 以 下 三 行 代 码 。 


<%@ Control Language="C#" RutoEventWireup="true"” CodeBehind="Header. 
SEES ES 

Inherits="UserControlSample.UserControls .Header" %> 

<%@ Register src="Weather.ascx" tagname="Weather" tagprefix="ucl" $%> 
<ucl:Weather ID="Weatherl" runat="server" /> 


其 中 ed 指令 ， 说 明 这 是 一 个 用 户 控 件 ， 还 说 明了 控件 的 一 
基本 信息 .第 二 行 语句 是 Weather 用 户 控 件 的 注册 指令 ,在 当前 页 面 (控件 ) 中 注册 了 Weather 
用 户 控 件 。 es 语句 添加 了 一 个 Weather 控件 。 

(5) 修改 Headerascx 控件 源码 ， 使 其 包含 主要 功能 的 链接 ， 代 码 如 下 : 


<%@ Control Language="C#" RutoEventWireup="true"” CodeBehind="Header.ascx. 
cs" Inherits="MyWeb.UserControls.Header" $%> 
<s-- 注 册 天 气 预 报 用 户 控件 --$> 
<%@ Register src="Weather.ascx" tagname="Weather" tagprefix="ucl" 委 > 
<div style="background:#f2f2fe; width:1000px; text-align:left;"> 
<%-- 最 外 层 div--%> 
<div style="float:left"> <g%--LOGO， 向 左 浮动 --g%> 
<img src="../images/title.png" alt="" /> <span style="margin-left:180px; 
margin-right:30px;"> 
当前 日 期 : 
<% 一 -调用 C# 代 码 显示 当 前 日 期 和 时 间 --%> 
<%=DateTime .Now.ToLongDateSstring()+"gnbsp;"+DateTime.Today.DayOfWeek %> 
</span> 
</div> 
<ucl:Weather ID="Weatherl"” runat="server" /> <%-- 天 气 预 报 控件 --%> 
</div> 
<% 一 -以 下 为 页 面 顶 部 的 导航 条 ， 每 个 导航 项 目 包括 一 个 图 片 和 一 个 文字 --%> 
<div id="NavMenu"> 
<div class="ImageMenu"><img src="../images/user.png" alt=" 农 民 档 案 "/><br/> 
<a href="../People/FamilyPage.aspx"> 农 民 参 合 档案 </a></div> 
<div class="ImageMenu"><img src="../images/medicine.png"” alt=" 药 品目 录 " 
/><br/> 
<a href="../Medicine/MedicinePage.aspx"> 药 品 检查 目录 </a></div> 
<div class="ImageMenu"><img src="../images/hospital.png"” alt=" 医 疗 机 构 " 
/><br/> 
<a href="../Hospital/HospitalPage.aspx"> 定 点 医疗 机 构 </a></div> 
<div class="ImageMenu"><img src="../images/policy-.png"” alt=" 报销 政策 " 


/><br/> 

<a href="../Policy/CopensateRatePage.aspx"> 报 销 政策 管理 </a></div> 

<div class="ImageMenu"><img src="../images/dictionary.png”alt=" 字 典 维护 " 
/><be/> 


<a href="../Dictionary/DictionaryPage.aspx"> 数 据 字 典 维 护 </a></div> 
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<div class="ImageMenu"><img src="../images/lock.png" alt=" 用 户 权 限 " /><br/> 
<a href="../UserRight/ChangePassPage.aspx"> 用 户 权 限 管理 </a></div> 
</div> 


(6) 在 项 目 中 添加 页 脚 控 件 Footer.ascx， 页 脚 控 件 主要 包含 一 些 说 明 信 息 ， 代 码 如 下 : 


<%@ Control Language="C#" AutoEventWireup="true" CodeBehind="Footer.ascx. 
cs" Inherits="MyWeb.UserControls.Footer" %> 
<div style="width:1000px; height:30px; font-size:1llpx; color:#666; text-— 
align:center; 

border:dashed lpx silver; background:#f2f2f2; float:none; clear:both;"> 
版 权 所 有 : XXXXX 公司 。 地 址 : XX 省 XX 市 XXXXXX 街道 XX 号 。 联 系 电话 : 010-12345678 
<br /> 
技术 支持 : <a href="mailto:youremailegmail.com"> 给 我 发 邮件 </a></div> 


(7) 在 项 目 中 添加 一 个 页 面 XNH.aspx， 打 开 此 页 面 并 切换 到 设计 视图 ， 从 “解决 方 


案 资 源 管理 器 ”中 把 Header 和 Footer 控件 拖 动 到 页 面 中 。 页 面 关 键 代码 如 下 : 


<%@ Page Language="C#" AutoEgventWireup="true" CodeBehind="xnh.aspx.cs" 
Inherits="UserControlSample.xnh" StylesheetTheme="default" $%> 

<!-- 注 册 Header 控件 --> 

<%@ Register src="UserControls/Header.ascx" tagname="Header" tagprefix= 
bh i 4 

<!-- 注 册 Footer 控件 --> 

<%@ Register src="UserControls/Footer.ascx" tagname="Footer" tagprefix= 
人 入 六 


<body> 


<form id="forml" runat="server"> 
<div> 
<ucl:Header ID="Headerl" runat="server" /> <!-- 创 建 Header 控件 --> 


<div style="margin:3px; border:dashed lpx silver; height:100px;"> 
<h3> 这 里 是 正文 内 容 </h3> 
</div> 
<uc2:Footer ID="Footerl" runat="server" /> <!-- 创 建 Footer 控件 --> 
</div> 
</form> 

</body> 


(8) 运行 XNH.aspx 页 面 ， 运 行 结果 见 本 例 开 始 处 的 图 1.29。 


1.5.2 添加 自 定义 属性 


自 


前 面 所 介绍 的 Header 和 Footer 用 户 控件 都 不 与 页 面 上 的 其 他 控件 打交道 ， 是 比较 简 


单 的 用 户 控件 。 在 实际 应 用 中 ， 作 为 页 面 的 一 部 分 ， 用 户 控件 和 页 面 上 其 他 控件 需要 进行 


交流 。ASPNET 的 标准 控件 如 Button 等 ， 都 有 一 些 属 性 〈 如 Text) 和 事件 (如 Click)， 页 
面 代码 正 是 通过 这 些 属性 实现 与 控件 的 交流 和 通信 。 用 户 控 件 继承 自 
System.Web.UI.UserControl， 也 继承 了 很 多 通用 属性 和 事件 ， 但 这 些 属性 和 事件 通常 并 不 
能 满足 要 求 ， 这 时 就 需要 开发 人 员 自 行 定义 属性 和 事件 。 


【 例 1-20】 用 户 控件 的 属性 。 
本 例 将 开发 一 个 进度 条 控件 。 这 个 控件 有 三 个 自 定义 属性 : 当前 进度 、 背 景色 和 前 景 
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色 。 本 例 用 一 个 Table 表示 进度 条 ， 用 Table 中 的 单元 格 Cell 表示 进度 块 。 进 度 条 中 共有 
20 个 进度 块 ， 进 度 条 数值 范围 为 0 一 100， 每 5 个 数值 对 应 一 个 进度 块 。 

(1) 创建 一 个 ASPNET Web 应 用 程序 。 

(2) 在 项 目 中 添加 一 个 ASPNET 用 户 控件 ， 习 
个 服务 器 端 Table 控件 ， 代 码 如 下 : 


<%@ Control Language="C#" AutoEventWireup="true" CodeBehind="ProgressBar. 
ascx.cs" Inherits="UserControlSample.UserControls.ProgressBar" %> 
<asp:Table ID="tablel" runat="server" BorderWidth="1lpx" CellPadding="1" 
CellSpacing="1" Height="15px"” Width="200px"> 

</asp:Table> 


(3) 为 ProgressBar 控件 添加 一 个 BackColor 属性 ， 表 示 进 度 条 背景 色 。 代 码 如 下 : 


// 进 度 条 背景 色 
public Color backColor 
{ 
get 
i‘ 
// 如 果 ViewState 中 没有 保存 背景 色 ， 则 返回 默认 颜色 灰色 
if (ViewState["BackColor"] == nul1) 
return Color.Silver; 
// 如 果 Viewstate 中 已 经 保存 了 背景 色 ， 则 返回 保存 的 颜色 
return (Color)ViewState["BackColor"]; 


吓 


命名 为 ProgressBar， 并 在 其 中 添加 一 


ViewState["BackColor"] = value; // 把 新 颜色 保存 到 ViewState 


全 注意 : 由 于 Web 应 用 的 无 状态 性 ， 为 控件 添加 属性 时 需要 考虑 属性 的 持久 性 ， 通 常 可 
以 通过 视图 状态 ViewState 来 实现 。 


(4) 为 ProgressBar 控件 添加 前 景色 属性 ForeColor， 代 码 如 下 : 


// 进 度 条 前 景色 
public Color foreColor 
{ 
get 
{ 
// 如 果 ViewState 中 没有 保存 前 景色 ， 则 返回 默认 颜色 绿色 
if (ViewState ["ForeColor"] == null) 
return Color.Green; 
// 如 果 Viewstate 中 已 经 保存 了 前 景色 ， 则 返回 保存 的 颜色 


return (Color)ViewState["ForeColor"]; 


set 
{ 
ViewState["ForeColor"] = value; // 把 新 颜色 保存 到 ViewState 
有 
} 


(5) 为 ProgressBar 控件 添加 属性 Progess， 表 示 当 前 进度 。 进 度 条 的 最 大 进度 为 100， 
在 设置 属性 时 要 检测 值 的 合法 性 。 代 码 如 下 : 
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// 当 前 进度 (有效 值 为 0 到 100) 
public int progress 
L 
get 
‘ 
// 如 果 ViewState 中 没有 保存 当前 进度 ， 则 返回 0， 否 则 返回 保存 的 进度 
if (ViewState["Progress"] == null) 
return 0; 
return (int)ViewState["Progress"]; 
} 


set 


// 如 果 要 设置 的 值 合法 〈 在 0 到 100 之 间 ) ， 则 接受 ， 否 则 拒绝 
if (value > 100 || value < 0) return; 
ViewState["Progress"] = value; 


(6) 在 ProgressBar 控件 的 PreRender 事件 中 ， 根 据 前 景色 、 背 景色 、 当 前 进度 的 值 ， 
生成 进度 条 控件 界面 。 代 码 如 下 : 


protected void Page Load (object sender, EventArgs e) 
{ ”// 为 控件 的 PreRender 事件 添加 事件 处 理 程序 
this.PreRender += new EventHandler (ProgressBar PreRender); 
. 
void ProgressBar PreRender (object sender, EventArgs e) 
{ ”// 获 得 当前 进度 所 对 应 的 进度 块 (一块 表示 5 个 数值 7 
int blocks=(int)Math.Round (progress*20.0/100.0); 
TableRow row = new TableRow(); // 创 建 Table 中 的 一 行 
// 添 加 单元 格 ， 一 共 添 加 20 个 
FOr (Int i = OF 1 <20 3. 444) 
{ 
TableCell cell = new TableCell (); 
coll-. Toxt = mn 
// 根 据 当 前 进度 设置 单元 格 颜色 
LE < Dlocksy 
cell.BackColor = foreColor; 
else 
cell.BackColor = backColor; 
row.Cells.Add (cell); // 把 单元 格 添加 到 行 
} 
tablel .Rows .Add (row); 
} 


(7) ProgressBar 控件 到 此 为 止 创建 完成 ， 下 面 编写 一 个 测试 页 面 。 在 项 目 中 添加 一 个 
新 的 ASPNET 页 面 ProgressPage.aspx， 把 ProgressBar 控件 拖 到 页 面 上 ， 并 在 页 面 上 放置 
按钮 等 控件 以 进行 测试 。ProgressPage.aspx 关键 代码 如 下 : 


<form id="forml" runat="server"> 
<div> 
<ucl :ProgressBar ID="ProgressBarl" runat="server" /> 
<$ 一 -进度 条 用 户 控 件 --%> 
<br > 
<asp:Button ID="Buttonl" runat="server"” Text=" 增 加 " onclick="Buttonl 
A le 
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<asp:Button ID="Button2" runat="serVer" Text=" 减 少 " onclick="Button2 


Cliok /><bre /> 
前 景色 : 


<asp:DropDownList ID="foreColorList" runat="server"> 


<%-- 前 景色 下 拉 列 表 --%> 


<asp:ListItem Value="Green"> 绿 色 </asp:ListItem> 
<asp:ListItem Value="Red"> 红 色 </asp:ListItem> 
<asp:ListItem Value="Blue"> 蓝 色 </asp:ListItem> 
</asp:DropDownList> 
背景 色 ; 


<asp:DropDownList ID="backColorList" runat="server"> 


<g%-- 背 景色 下 拉 列表 --%> 


<asp:ListItem Value="Black"> 黑 色 </asp:ListItem> 
<asp:ListItem Value="White"> 白 色 </asp:ListItem> 


<asp:ListItem Value="Silver"> 灰 色 </asp:ListItem> 
</asp:DropDownList> 


<asp:Button ID="Button3" runat="server"” Text=" 修 改 颜 色 " onclick= 


"Button3 Click" /> 
</div> 
</form> 


(8) 为 ProgressPage.aspx 页 面 的 3 个 按钮 编写 代码 如 下 : 


// 增 加 进度 ， 出 于 演示 目的 ， 为 当前 进度 添加 一 个 随机 数 
protected void Button1l Click(object sender, EventArgs e) 
{ 
Random r = new Random(DateTime.Now.Second); 
ProgressBarl.progress += r.Next (20); 
} 
// 减 少 进度 ， 出 于 演示 目的 ， 为 当前 进度 添加 一 个 随机 数 
protected void Button2 Click(object sender, EventArgs e) 
{ 
Random r = new Random(DateTime.Now.Second); 
ProgressBarl.progress -= r.Next (20); 


// 改 变 进度 条 颜色 

protected void Button3 Click(object sender, EventArgs e) 

{ 
Color fore = Color.FromName (foreColorList.SelectedValue); 
Color back = Color.FromName (backColorList.SelectedValue); 
ProgressBarl.foreColor = fore; 
ProgressBarl.backColor = back; 


有 


(9) 运行 ProgressPage.aspx 页 面 ， 运 行 界 面 如 图 1.31 所 示 。 


文件 中 忽 各 中 ”查看 W 历史 芒 ) 书签 加。 工具 中 各 动 中 
CO - © /ban 


Googke | nrolID -| +e- -» 


http:f/locdhesmeressTaee aspz| 二 | - 


增加 | _ 减少 
前 景色 :| 气色 司 背景 色 : [6 色 司 有 可 


[及 EdE3 


图 1.31 进度 条 控件 示例 
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1.5.3 添加 自 定 义 事件 


ASPNET 标准 控件 都 有 各 种 事件 ， 如 Button 控件 有 Click 等 事件 。 控 件 通过 事件 机 制 
将 自身 的 状态 变化 通知 外 界 环境 。 对 于 用 户 控件 来 说 ， 很 多 时 候 也 需要 事件 机 制 与 外 界 进 
行 通信 。 

【 例 1-21】 用 户 控件 的 事件 。 

本 例 将 制作 一 个 可 以 复 用 的 登录 用 户 控件 , 该 控件 的 布局 类 似 于 ASPNET 标准 控件 中 
的 Login 控件 , 包含 两 个 文本 框 和 一 个 “登录 ”按钮 。 用户 输入 用 户 名 和 密码 后 ， 单 击 “ 登 
录 ” 按 钮 ， 即 可 尝试 登录 。 
于 这 是 一 个 可 以 复 用 的 控件 ， 可 以 被 应 用 于 不 同 的 程序 中 ， 所 以 控件 的 开发 者 并 不 
知道 如 何 验证 用 户 名 和 密码 ， 例 如 ， 如 果 用 户 信息 保存 在 数据 库 中 ， 那 么 控件 开发 者 不 知 
道 数据 库 类 型 、 表 名 、 列 名 等 ， 从 而 无 法 对 登录 信息 的 合法 性 进行 验证 。 具 体 的 验证 工作 
应 该 由 控件 的 使 用 者 而 不 是 开发 者 来 完成 。 所 以 , 控件 的 开发 者 不 能 在 “登录 ”按钮 的 Click 
事件 中 写 代码 验证 登录 ， 而 是 需要 把 这 个 验证 交 给 控件 所 在 的 页 面 来 做 ， 这 就 需要 定义 一 
个 事件 来 完成 这 个 功能 。 

(1) 创建 一 个 ASPNET 应 用 程序 。 

(2) 在 项 目 中 添加 一 个 用 户 控件 ， 命 名 为 MyLogin， 并 在 其 中 放置 用 于 输入 用 户 名 和 
密码 的 TextBox， 一 个 “登录 ”Button， 以 及 两 个 验证 控件 。 为 了 使 各 个 控件 排列 整齐 ， 使 
用 table 布局 。 代 码 如 下 : 


<table border="0" cellpadding="1" cellspacing="0" style="border-collapse: 
collapse;"> 
<tr><td align="center" colspan="2"> 登 录 </td> </tr> 
<tr><td align="right"> 
<asp:Label ID="UserNameLabel" runat="server" AssociatedControlID= 
"UserName"> 用 户 名 :</asp:Label></td> 
<td> <asp:TextBox ID="UserName" runat="server"></asp:TextBox> 
<asp:RequiredFieldValidator ID="UserNameRequired" runat="server" 
ControlToValidate="UserName" ErrorMessage=" 必 须 填 写 " 用 户 名 "。" ToolTip=" 
必须 填写 用 户 名 。" ValidationGroup="Login1">*</asp:RequiredFieldValidator> 
</td> 
<hEr> 
<tr><td align="right"> 
<asp:Label ID="PasswordLabel" runat="server" AssociatedControlID= 
"Password"> 密 码 :</asp:Label></td> 
<td> 
<asp:TextBox ID="Password" runat="server" TextMode="Password"></asp: 
TextBox> 
<asp:RequiredFieldValidator ID="PasswordRequired" runat="server" 
ControlToValidate="Password"” ErrorMessage=" 必 须 填写 "密码 "。" ToolTip=" 
必须 填写 密码 。" ValidationGroup="Loginl1">*</asp:RequiredFieldValidator> 
</td> 
</Er 
<tr> <td align="center" colspan="2" style="color:Red;"> 
<asp:Literal ID="FailureText" runat="server" EnableViewState= 
"False"></asp:Literal> </td> 


</tr> 
<tr><td align="right" colspan="2"> 
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<asp:Button ID="LoginButton" runat="server" CommandName="Login" 
Text=" 登 录 " 
ValidationGroup="Loginl" onclick="LoginButton Click" /></td> 
< /Er> 
</table> 


控件 在 设计 状态 下 的 外 观 如 图 1.32 所 示 。 


图 1.32 登录 控件 设计 界面 


(3) 根据 前 面 的 讨论 , MyLogin 用 户 控件 需要 通过 事件 通知 其 外 部 页 面 。 事 件 发 生 时 ， 
需要 把 用 户 名 和 密码 作为 事件 参数 传递 给 外 部 页 面 。 因 此 ， 需 要 在 项 目 中 添加 一 个 类 ， 描 
述 事 件 参数 。 该 类 从 EventArgs 继承 ， 有 name 和 password 两 个 属性 ， 代 码 如 下 : 


// 登 录 事 件 参数 
public class MyLoginEventArgs : EventArgs 


{ 
public string name { get; set; } 
public string password { get; set; } 
和 


(4) 在 MyLogin.ascx 用 户 控件 中 声明 一 个 事件 myLoginEvent， 代 码 如 下 : 

public event EventHandler<MyLoginEventArgs> myLoginEvent; // 定 义 登录 事件 

(5) 在 MyLogin.aspx 用 户 控 件 “ 登 录 ” 按 钮 的 Click 事件 中 ,触发 自 定义 的 用 户 事件 。 
代码 如 下 : 


protected void LoginButton Click(object sender, EventArgs e) 
{ 


/ /如果 登录 事件 附加 了 事件 处 理 程序 
if (myLoginEvent != null) 


// 创 建 事件 参数 并 设置 其 值 

MyLoginEventArgs arg = new MyLoginEventArgs(); 
arg.name = UserName.Text; 

arg.password = Password.Text; 


myLoginEvent (this, arg); // 调 用 事件 处 理 程序 
: 
(6) 在 MyLogin.aspx 用 户 控件 中 添加 两 个 方法 ， 用 于 显示 和 清除 错误 提示 信息 。 代 码 
如 下 : 


// 显 示 登 录 错误 信息 
//<param name="message"> 要 显示 的 错误 信息 </param> 
public void showFailure(string message) 
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FailureText.Text = message; 
FailureText .Visible = true; 


// 清 除 错误 信息 

public void clearFailure() 
FailureText.Text = ""; 
FailureText .Visible = false; 


(7) 在 项 目 中 添加 一 个 ASPNET 页 面 MyLoginPage.aspx， 用 于 测试 用 户 控 件 。 页面 代 
码 如 下 : 


<form id="forml" runat="server"> 
<div> 
<ucl:MyLogin ID="MyLoginl" runat="server" /> 
</div> 
</form> 


(8) 在 MyLoginPage.aspx 页 面 中 ， 为 MyLogin 用 户 控件 的 myLoginEvent 事件 添加 事 
件 处 理 程序 ， 以 验证 用 户 名 和 密码 正确 性 。 代 码 如 下 : 


protected void Page Load (object sender，EventRrgs e) 


// 为 登录 用 户 控件 的 myLoginEvent 事件 添加 事件 处 理 程序 

MYLogin1l1.myLoginEvent += new EventHandler 

<UserControlSample.UserControls.MyLoginEventArgs> 
(MyLoginl myLoginEvent); 


} 

// 登 录用 户 控 件 的 myLoginEvent 事件 处 理 程序 

void MyLoginl myLoginEvent (object sender, 

UserControlSample.UserControls.MyLoginEventArgs e) 

{ 
if (e.name == "admin" && e.password == "abcd") // 检 测 用 户 名 和 密码 
{ 

Response.Redirect ("LoginSuccess.aspx"); // 如 果 正 确 则 转 到 登录 成 功 页 面 

} 


else 
{ 
MyLogin1 .showFailure ("用 户 名 或 密码 错误 ") ; // 如 果 错 误 则 显示 错误 提示 
} 


(9) 运行 MyLoginPage.aspx 页 面 ， 运 行 界面 如 图 1.33 所 示 。 


myLegin 用 户 控件 ~ Noxills Ff 有 口 | x| 
文件 下) 编辑 邓 ) Er PR 书本 四 工具 D 和 
< ERERcealcs 


a 
CT SA 
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图 1.33 登录 控件 运行 界面 
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1.6 自 定义 控件 


与 用 户 控 件 类 似 , 自 定义 控件 也 是 开发 人 员 自 行 设计 开发 的 可 以 在 ASPNET 页 面 上 使 
用 的 控件 。 但 是 自 定义 控件 与 用 户 控件 相 比 ， 也 有 明显 的 区 别 ， 二 者 的 创建 、 使 用 过 程 都 
不 相同 ， 自 定义 控件 的 开发 往往 比 用 户 控 件 更 难 ,功能 更 强大 ， 更 能 体现 开发 人 员 的 水 平 。 


1.6.1 自 定义 控件 概述 


自 定义 控件 功能 强大 , 其 行为 类 似 于 ASPNET 标准 控件 。 一 个 设计 良好 的 自 定 义 控 件 
可 以 复 用 于 各 个 项 目 中 ， 完 成 一 定 功能 ， 提 高 开发 效率 。 现 在 网 上 有 较 多 优秀 的 ASPNET 
自 定义 控件 ， 如 Infragistics NetAdvantage、Xceed Chart、Teleric、ComponentArt 等 。 

ASPNET 自 定 义 控件 通常 是 在 一 个 单独 的 .NET 项 目 中 ， 经 过 编译 后 生成 一 个 程序 集 
(DLL 文件 )。 使 用 自 定 义 控 件 的 开发 人 员 需 要 在 项 目 中 添加 对 这 个 程序 集 的 引用 ， 还 可 以 
把 自 定 义 控件 拖 动 到 Visual Studio 开发 环境 的 工具 箱 里 , 然后 像 使 用 ASPNET 标准 控件 一 
样 使 用 自 定义 控件 。 对 于 控件 使 用 者 来 说 ， 自 定义 控件 与 ASPNET 标准 控件 在 使 用 上 很 
相似 。 

与 用 户 控 件 相 比 ， 自 定义 控件 功能 更 强大 ， 使 用 更 方便 ， 开 发 更 复杂 ， 可 以 作为 单独 
的 DLL 文件 进行 发 布 。 对 于 一 个 ASPNET 开发 人 员 来 说 ， 自 定义 控件 比 用 户 控 件 更 能 体 
现 开发 人 员 的 思路 和 水 平 。 国内 已 经 有 一 些 程序 员 开 发 出 了 免费 ASPNET 自 定义 控件 ， 
被 广泛 使 用 ， 如 用 于 数据 分 页 的 AspNetPager 控件 。 

一 个 ASPNET 自 定义 控件 对 应 于 一 个 类 ， 这 个 类 通常 从 System.Web.UI.Control 或 
System.Web.UI.WebControls.WebControl 派生 。 如 果 自 定义 控件 没有 用 户 界面 ， 那 么 一 般 从 
Control 类 派生 ， 如 果 自 定义 控件 有 用 户 界面 ， 那 么 一 般 从 WebControl 类 派生 。 


1.6.2 创建 和 使 用 简单 的 自 定义 控件 


大 多 数 实用 的 自 定义 控件 都 较为 复杂 ， 需 要 考虑 状态 保持 、 回 发 、 事 件 等 各 种 因素 。 
对 于 没有 控件 开发 基础 的 人 来 说 ， 需 要 循序 渐进 地 学 习 和 练习 。 下 面 将 开发 一 个 最 简 的 
HelloWorld 控件 ， 以 此 来 说 明 创建 和 使 用 自 定义 控件 的 过 程 。 

【 例 1-22】 HelloWorld 控件 。 

本 例 将 开发 一 个 HelloWorld 控件 , 此 控件 的 唯一 作用 就 是 在 页 面 上 显示 “Hello, World” 
文字 。 

(1) 创建 一 个 C# 类 库 项 目 ， 命 名 为 CustomControls， 注 意 不 要 创建 成 ASPNET 项 目 。 
正确 操作 方法 如 图 1.34 所 示 。 

(2) 在 项 目 中 添加 一 个 HelloWorld 类 。 

(3) 本 例 中 ，HelloWorld 类 需要 从 System.Web.UI.WebControls.WebControl 类 派生 ， 
WebControl 类 在 System.Web 程序 集中 。 在 默认 情况 下 ， 类 库 项 目 并 不 包含 System.Web 程 
序 集 ， 需 要 手动 添加 对 此 程序 集 的 引用 。 操 作 步 又 是 在 项 目 上 右 击 ， 从 弹出 的 快捷 菜单 中 


nj 
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选择 “添加 引用 ”命令 。 在 弹出 的 “添加 引用 ”对 话 框 中 找到 System.Web， 单 击 “ 确 定 ” 


按钮 ， 如 图 1.35 所 示 。 


ET Frmerork 4 。。”| 兰 序 限 据 :| 默认 全 | | S| ss Cesar p| 
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Reportine 
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a 粒 “WPF 上 自 定义 控件 库 
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图 1.34 添加 类 库 项 目 
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图 1.35 添加 对 System.Web 的 引用 


添加 System.Web 引用 后 ， 修 改 HellowWorld 类 代码 ， 使 其 从 WebControl 类 派生 。 代 


但 如 下 : 
public class HelloWorld:WebControl 
} 


(4) 要 让 控件 在 页 面 上 显示 出 来 ， 就 是 让 控件 在 页 面 上 生成 HIML 代码 。 可 以 通过 如 


山 


写 WebControl 类 的 Render0 方 法 向 页 面 输出 任意 HTML 代码 。 本 例 需 要 输出 “Hello World” 


字样 ， 相 应 代码 如 下 : 


protected override void Render (System.Web.UI.HtmlTextWriter writer) 
: 


writer.Write("<hr/>"); 


writer.Write("<span style="'font-weight:bold;'>Hello,World.</span>"); 


writer.Write("<hr/>"); 


as Sa 


第 1 章 ASPNET 网 络 开发 基础 


(5) 编译 CustomControls 项 目 ， 则 在 项 目 文件 夹 下 的 Bin 目录 中 生成 一 个 与 项 目 名 称 
相同 的 dll 文件 。 到 此 为 止 一 个 最 简单 的 自 定义 控件 就 开发 完成 了 。 下 面 介绍 如 何在 
ASPNET 项 目 中 使 用 此 控件 。 

(6) 新 建 一 个 ASPNET Web 应 用 程序 ， 命 名 为 
ControlTest， 此 项 目 是 自 定义 控件 的 测试 项 目 。 得 

(7) 如 果 刚 才 所 创建 的 测试 项 目 ControlTest 与 控件 项 Haecontralstate 
目 CustomControls 在 同一 解决 方案 中 , 则 打开 Visual Studio 
工具 箱 ,可 以 看 到 自动 添加 了 一 个 CustomControls 选项 卡 ， 
选项 卡 中 有 一 个 HelloWorld 控件 。 选 项 卡 的 名 字 就 是 控件 
项 目的 名 字 ， 选 项 卡 上 控件 的 名 字 就 是 控件 类 的 名 字 ， 如 
图 1.36 所 示 。 

如 果 刚 才 所 创建 的 测试 项 目 ControlTest 与 控件 项 目 
CustomControls 不 在 同一 解决 方案 中 ， 那 么 需要 手工 在 工 


HelloyiewState 
Hello¥orld 


PETTITT 


国 Ajax Contorl Toolkit 


具 箱 中 添加 自 定义 控件 。 有 具体 操作 步骤 如 下 。 已 % 枫 


(a) 把 控件 项 目 编译 后 生成 的 CustomControls.dll 文件 
复制 到 磁盘 任意 位 置 。 

(b) 在 Visual Studio 开发 环境 工具 箱 空白 处 右 击 ， 从 弹出 的 快捷 菜单 中 选择 “添加 选 
项 卡 ” 选 项 ， 给 新 添加 的 选项 卡 任意 命名 ， 如 MyControl。 

(c) 打开 新 添加 的 选项 卡 , 在 选项 卡 空白 处 空白 处 右 击 , 从 弹出 的 快捷 菜单 中 选择 “ 选 
择 项 ” 选项， 在 打开 的 对 话 框 中 选择 刚才 所 复制 的 CustomControls.dll 文件 。 这 样 就 在 工具 
箱 里 出 现 了 HelloWorld 自 定义 控件 。 

(8) 在 ControlTest 项 目 中 ,打开 一 个 页 面 ， 从 工具 箱 里 把 HelloWorld 控件 拖 动 到 页 面 
上 ， 此 时 页 面 关键 代码 如 下 : 


<%@ Register Assembly="CustomerControls" Namespace="CustomerControls" 
TagPrefix="ccl" %> 


图 1.36 工具 箱 中 的 自 定义 控件 


<body> 
<form id="forml" runat="server"> 
<div> 
<ccl:HelloWorld runat="server" id="hellol" /> 
</div> 
</form> 
</body> 


上 述 代 码 中 ，<%@Register 一 行 代码 注册 了 ITT lolx 
HelloWorld 控件 ，<ccl:HelloWorld 一 行 代码 使 用 了 文旦 W 编 入 如 ”查看 WD 历史 (G) 书签 的 工具 人 ) 
该 控件 。 < mae 


(9) 运行 此 页 面 ， 运 行 界面 如 图 1.37 所 示 。 ee CE 
(10) 查看 页 面 HTML 源码 ， 可 以 看 到 如 下 内 | | http://localhos™5/Default. aspx =- 


容 , 这 些 内 容 正 是 在 HelloWorld 控件 中 所 写 的 代码 。 各 过 近 人 全 


<hr/> 自 定义 控件 示例 <br/><span style= 不 通 [三 ESIE 
'font-weight:bold; '>Hello, World. 
</span><hr/> 图 1.37 HelloWorld 控件 运行 界面 
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1.6.3 添加 属性 


1.6.2 节 所 讲 的 HelloWorld 控件 , 只 能 显示 固定 的 内 容 , 控件 使 用 者 无 法 定义 控件 的 状 
态 ， 如 其 中 的 文字 、 字 体 、 颜 色 等 。 要 想 实现 对 控件 更 加 细致 的 控制 ， 就 需要 用 到 自 定义 


属性 。 下 面 通过 一 个 例子 说 明 如 何 为 自 定义 控件 添加 属性 。 
【 例 1-23】 自 定义 控件 属性 。 


本 例 将 设计 一 个 简单 控件 ， 此 控件 在 页 面 上 输出 一 条 问候 语句 ， 如 “你 好 张 三 ” 


中 人 名 、 字 体 颜色 都 可 以 设置 。 
(1) 新 建 一 个 类 库 项 目 CustomerControls (也 可 使 用 例 1-22 所 创建 的 项 目 )。 


(2) 在 CustomerControls 项 目 中 添加 对 System.Web 和 System.Color 程序 集 的 引 上 
体操 作 步 又 与 例 1-22 相同 。 


(3) 在 CustomerControls 项 目 中 添加 一 个 继承 自 WebControl 类 的 Hello 类 , 代码 如 下 : 


public class Hello:WebControl 


(4) 从 WebControl 派生 的 自 定义 控件 默认 在 页 面 上 呈现 为 一 个 span 元 素 ， 即 控件 的 
最 外 层 标记 为 span， 控 件 内 容 都 在 这 个 标记 中 。 本 例 将 改变 这 个 默认 行为 ， 用 一 个 div 元 


素 显示 自 定义 控件 ， 这 需要 重 写 TagKey 属性 ， 如 下 代码 所 示 。 


// 将 自 定义 控件 的 HTML 标记 设置 为 div (默认 为 span) 
protected override HtmlTextWriterTag TagKey 
| 

get 

{ 

return HtmlTextWriterTag.Div; 

} 

} 


(5) 在 Hello 类 中 ， 添 加 3 个 属性 ， 分 别 描述 姓名 和 两 种 颜色 。 代 码 如 下 : 


public string name { get; set; } // 你 好 后 面 的 姓名 
public Color helloColor { get; set; } //“ 你 好 ”的 颜色 
public Color nameColor { get; set; } // 姓 名 的 颜色 


(6) 在 Hello 类 中 重 写 RenderContents(0) 方 法 ， 按 照 指定 颜色 输出 问候 语 。 代 码 如 下 : 


// 输 出 控件 内 容 
protected override void RenderContents (HtmlTextWriter writer) 
{ 

string cl1=ColorTranslator .ToHtml (helloColor); 


// 将 .NET 的 颜色 转换 为 HTML 形式 


// 添 加 颜色 样式 ， 此 语句 将 生成 以 下 代码 : ”style="color: 颜 色 " 


writer.AddAttribute (HtmlTextWriterAttribute.Sstyle, "color:" + cl1); 


writer.RenderBeginTag (HtmlTextWriterTag.Span); 


// 输 出 一 个 span 开始 标记 <span> 


writer.Write ("你 好 "); // 输 出 你 好 
writer.RenderEndTag (); // 结 束 前 一 个 HTML 标记 ， 即 span 


string c2=ColorTranslator.ToHtm]l (nameColor); 


// 以 下 语句 以 特定 颜色 在 span 中 输出 姓名 


writer.AddAttribute (HtmlTextWriterAttribute.Sstyle, "color:" + c2); 
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writer.RenderBeginTag (HtmlTextWriterTag.Span); 
writer.Write (name); 
writer.RenderEndTag (); 
} 
(7) 编译 CustomerControls 项 目 以 生成 dl 文件 。 
(8) 新 建 一 个 ASPNET Web 应 用 程序 TestControl 作为 控件 的 测试 项 目 〈 也 可 以 使 用 
例 1-22 所 建 的 项 目 )。 
(9) 在 TestControl 项 目 中 引用 CustomerControls.dll 文件 ， 并 把 控件 放 到 工具 箱 中 。 具 
体操 作 步 又 与 例 1-22 所 述 相同 。 
(10) 在 TestControls 项 目 中 添加 一 个 页 面 ， 将 Hello 控件 拖 到 页 面 上 ， 并 设置 控件 的 
3 个 属性 。 页 面 关 键 代 码 如 下 : 
< 注册 自 定义 控件 -一色 


<%@ Register Assembly="CustomerControls" Namespace="CustomerControls" 
TagPrefix="ccl" 多 > 


<body > 

<form id="forml" runat="server"><div> 

<%-- 放 置 一 个 自 定义 控件 Hello， 并 设置 三 个 属性 --%> 

<ccl:Hello runat="server" id="hello2" name=" 赵 钱 孙 " helloColor="Blue" 
nameColor="Green"/> 

</div></form> 
</body> 


(11) 运行 上 一 步 所 添加 的 页 面 ， 可 以 看 到 ， 在 
浏览 器 中 以 指定 的 颜色 显示 了 问候 语 。 运 行 界面 如 Neen/Aecshbesws/pefont espr | 


图 1.38 所 示 。 你 好 赵 钱 孙 
1.6.4 ”状态 保持 概述 [天 El 本 


图 1.38 自 定义 控件 属性 示例 

HTTP 协议 是 一 种 无 状态 协议 ， 基 于 HTTP 协议 的 Web 应 用 程序 默认 情况 下 也 是 无 状 
态 的 ， 这 给 Web 应 用 的 开发 造成 不 便 。 各 种 Web 开发 工具 (包括 ASPNET) 都 采用 一 定 
的 方式 实现 状态 保持 ， 如 Session 等 。 在 开发 自 定义 控件 时 ， 如 果 考 虑 不 到 实现 状态 保持 
功能 , 那么 控件 的 运行 就 会 出 现 意 想不到 的 错误 , 例如 , 把 某 一 控件 的 Text 属性 设置 为 “ 测 
试 ” 经 过 页 面 回 发 以 后 ， 这 个 控件 的 Text 属性 就 会 恢复 成 原来 的 值 。 下 面 通过 一 个 例子 
来 演示 这 种 情况 。 

【 例 1-24】 有 状态 和 无 状态 。 

本 例 对 比 ASPNET 标准 控件 (以 Label 为 例 ) 和 1.6.3 节 开 发 的 自 定义 控件 Hello 在 状 
态 保持 方面 的 区 别 ， 以 说 明 状 态 保持 的 概念 和 作用 。 

(1) 创建 一 个 ASPNET Web 应 用 程序 。 

(2) 添加 一 个 新 的 页 面 Stateless.aspx， 在 页 面 上 放置 一 个 Label 及 一 个 自 定义 控件 
Hello， 然 后 再 放置 一 个 TextBox 和 两 个 Button， 页 面 关键 代码 如 下 : 


<form id="forml" runat="server"> 
<div> 
ASP.NET 标准 控件 (以 下 面 的 Label 为 例 ) 能 够 保持 状态 <br /> 
<asp:Label ID="Labell" runat="server" Text="Label" BorderStyle= 
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"Solid"></asp:Label><br /> 

自 定义 控件 Hello 不 能 保持 状态 <br /> 
<ccl:Hello runat="server" ID="hello"” name=" 初 始 值 " helloColor="Blue" 
nameColor="Green" BorderStyle="Solid"/> 


输入 新 值 : <asp:TextBox ID="TextBoxl" runat="server"></asp:TextBox> 


<asp:Button ID="Buttonl" runat="server"” Text=" 设 置 新 值 "” onclick= 
"Buttonl Click" /> 


<asp:Button ID="Button2"” runat="server"” Text=" 回 发 " /> 
</div> 
</form> 


(3) 当 单 击 “ 设 置 新 值 ”Button 时 ， 将 会 重新 设置 Label 和 Hello 控件 上 的 文本 ， 如 下 
代码 所 示 。 
protected void Button1l Click(object sender，EventRrgs e) 
{ 
Label1.Text = TextBoxl.Text; // 为 Label 控件 设置 新 的 文本 
hello.name = TextBoxl.Text;  // 为 Hello 控件 设置 新 的 姓名 


} 

(4) 页 面 上 “ 回 发 ”Button 的 作用 是 引起 页 面 回 发 ， 从 而 检验 控件 能 否 保 持 状态 。 由 
于 ASPNET 的 服务 器 端 Button 控件 默认 情况 下 将 引起 回 发 ， 所 以 不 需要 为 此 Button 编写 
事件 处 理 程序 。 

(5) 运行 Stateless.aspx 页 面 ， 在 文本 框 中 输入 新 的 文字 ， 单 击 “ 设 置 新 值 ”按钮 ， 可 
以 看 到 ，Label 控件 和 Hello 控件 的 文字 都 发 生 了 变化 。 此 时 ， 再 单 击 “ 回 发 ”按钮 引起 一 
次 页 面 回 发 过 程 ， 当 页 面 重新 加 载 后 ， 可 以 看 到 ，Label 文本 仍然 是 新 的 文字 ， 而 Hello 上 
的 文本 已 经 变 成 了 原来 的 文本 ， 即 Label 能 够 记忆 并 且 保持 其 文本 内 容 ， 而 Hello 则 不 能 ， 
如 图 1.39 所 示 。 


=|D|x 
文件 编辑 下 ) 查看 WD 历史 G) 书签 @@) 工具 CD) 帮助 人 D 


-一 Cx | b/c 

了 ] 和 类 -> 二 
hatesy/aaaalhezrmrStatelasxiaxgx | EI 
ASP. NET 标 准 控件 《以 下 面 的 Label 为 例 ) 能 够 保持 状态 


图 1.39 有 状态 和 无 状态 对 比 示例 


1.6.5 视图 状态 ViewState 


ASPNET 标准 控件 主要 采用 两 种 机 制 实现 状态 保持 功能 : 视图 状态 〈ViewState) 和 控 
件 状 态 〈ControlState )。 在 开发 自 定义 控件 时 通常 也 使 用 这 两 种 机 制 。 

ViewState 属性 提供 一 个 字典 对 象 ， 用 于 在 对 同一 页 的 多 个 请 求 之 间 保 留 值 。 这 是 页 
用 来 在 往返 行程 之 间 保 留 页 和 控件 属性 值 的 默认 方法 。 在 处 理 页 时 ， 页 和 控件 的 当前 状态 
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会 散 列 为 一 个 字符 串 ， 并 在 页 中 保存 为 一 个 隐藏 域 或 多 个 隐藏 域 (如 果 存 储 在 ViewState 
属性 中 的 数据 量 超过 了 MaxPageStateFieldLength 属性 中 的 指定 值 )。 当 将 页 回 发 到 服务 器 
村 ， 页 会 在 页 初始 化 阶段 分 析 视 图 状态 字符 串 ， 并 还 原 页 中 的 属性 信息 。 

在 浏览 器 中 浏览 一 个 ASPNET 页 面 时 ， 从 浏览 器 菜单 中 查看 页 面 的 HTML 源 代码 ， 
会 看 到 在 form 中 有 一 个 名 称 为 “ VIEWSTATE” 的 隐藏 域 (Hidden)， 其 中 的 内 容 是 一 些 
不 可 读 的 字符 ， 这 就 是 被 编码 后 的 视图 状态 。 下 面 是 一 个 页 面 中 的 ViewState 内 容 。 


<input type="hidden" name=" VIEWSTRTE" id="_ VIEWSTRATE" value= 
"/wEPDwUKMTc20DM1IMzg1OA9kFgICAw9kFgICAQ8PFgIleBFRleHQFCHJ1ZHNmZHNmZGRKta 
RgeiFCG54vjJxWMYiYGkvCGbY=" /> 


ViewState 是 一 个 System.Web.UI.StateBag 类 的 实例 ， 可 以 通过 System.Web.UI.Control 
类 的 ViewState 属性 得 到 一 个 控件 或 者 页 面 的 ViewState。ViewState 属性 声明 如 下 : 


protected virtual StateBag ViewState { get; } 


StateBag 是 一 个 类 似 于 字典 Dictionary 的 类 ， 可 以 通过 一 个 字符 类 型 的 关键 字 为 索引 
得 到 与 之 相对 应 的 内 容 。 如 果 没 有 与 关键 字 匹 配 的 内 容 ， 则 返回 null。 

【 例 1-2S】 控件 的 视图 状态 。 

本 例 演示 通过 视图 状态 机 制 保持 控件 的 属性 。 

(1) 创建 一 个 类 库 项 目 。 

(2) 在 项 目 中 添加 一 个 类 HelloViewState， 此 类 功能 与 例 1-23 中 的 Hello 控件 类 似 ， 
也 是 根据 指定 的 颜色 和 姓名 显示 一 条 问候 信息 。 所 不 同 的 是 ， 这 个 控件 使 用 了 ViewState 
保持 控件 属性 。HelloViewState 类 代码 如 下 : 


// 自 定义 控件 示例 ， 通 过 ViewState 保持 状态 
public class HelloViewState : WebControl 


{ 
#region 自 定义 属性 ， 通 过 ViewState 保存 这 些 属性 
// 下 面 3 个 常量 为 3 个 属性 在 ViewState 中 的 键 值 
private const string ViewKeyName = "nameviewstate"; 
private const string ViewKeyColorl "hellocolor"; 
private const string ViewKeyColor2 "namecolor™; 
// 你 好 后 面 的 姓名 
public string name 


! 


get 
| 


return ViewState [ViewKeyName] as string; 


ViewState [ViewKeyName] = value; 


} 
//“ 你 好 ”的 颜色 
public Color helloColor 
. 
Ee 
{ 
if (ViewState [ViewKeyColor1]==nul1) 
return Color.Green; 
return (Color)ViewState[ViewKeyColorl1]; 
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set 
1 
ViewState[ViewKeyColor]1] = value; 
3 
} 
//“ 姓 名 ”的 颜色 
public Color nameColor 
Li 
get 
{ 
if (ViewState [ViewKeyColor2] == null) 


return Color .Green7 
return (Color)ViewState[ViewKeyColor2]; 


ViewState[ViewKeyColor2] = value; 
} 
} 


#endregion 
// 将 自 定义 控件 的 HTML 标记 设置 为 div (默认 为 span) 
protected override HtmlTextWriterTag TagKey 


{ 
get 
' 


} 


return HtmlTextWriterTag.Div; 


protected override void RenderContents (HtmlTextWriter writer) 
1 
string cl = ColorTranslator.ToHtm]l (helloColor); 


// 将 .NET 的 颜色 转换 为 HTML 形式 
// 添 加 颜色 样式 ， 此 语句 将 生成 以 下 代码 : ”style="color :颜色 " 
writer.AddAttribute (HtmlTextWriterAttribute.Sstyle, "color:" + cl1); 
writer.RenderBeginTag (HtmlTextWriterTag.Span); 

// 输 出 一 个 span 开始 标记 <span> 


writer.Write ("你 好 "); // 输 出 你 好 
writer.RenderEndTag (); // 结 束 前 一 个 HTML 标记 ， 即 span 


// 以 下 语句 以 特定 颜色 在 span 中 输出 姓名 
string c2 = ColorTranslator.ToHtm] (nameColor); 
writer.AddAttribute (HtmlTextWriterAttribute.Style, "color:" + c2); 
writer.RenderBeginTag (HtmlTextWriterTag.Span); 
writer.Write (name); 
writer.RenderEndTag (); 
’ 


(3) 新 建 一 个 ASPNET 项 目 ， 在 项 目 中 添加 一 个 页 面 ControlViewState.aspx 以 测试 
HelloViewState 控件 。 页 面 关键 代码 如 下 : 


<form id="forml" runat="server"> 

<div> 

<ccl:HelloViewState runat="server" ID="controll" name=" 初 始 值 " helloColor= 

"Green" nameColor="Blue" /> 
输入 新 值 : <asp:TextBox ID="TextBoxl" runat="server"></asp:TextBox> 
<asp:Button ID="Button1" runat="server" Text=" 设 置 新 值 ” onclick= 
"Buttonl Click" /> 
<asp:Button ID="Button2" runat="server"” Text=" 回 发 " /> 

</div> 

</form> 


。S8 。 


第 1 章 ASPNET 网 络 开发 基础 


(4) ControlViewState.aspx.aspx 页 面 后 台 代 码 如 下 : 


protected void Button1l Click(object sender, EventArgs e) 
{ 


controll.name = TextBox].Text; 


. 


(5) 运行 ControlViewState.aspx.aspx 页 面 ， 设 置 新 的 姓名 并 回 发 页 面 ， 可 以 看 到 ， 所 
设置 的 新 姓名 能 够 保存 在 控件 中 。 运 行 界面 如 图 1.40 所 示 。 


Er CT Ez 历史 G) 书签 如 工具 中) Wm 
Ee < Ww FT 
CS 
Der: /flocalhos—YierState aspz| 二 | 司 
你 好 新 名 字 

输入 新 值 : 司 和 宁县 本 新 介 | 让 冯 | 


ES 二 


| 完成 


图 1.40 ViewState 保持 状态 


1.6.6 ”控件 状态 ControlState 


默认 情况 下 ，ASPNET 服务 器 端 控件 都 会 用 ViewState 进行 状态 保持 ， 这 样 就 会 在 
ASPNET 页 面 中 产生 很 大 的 隐藏 域 ， 影 响 服务 器 性 能 及 页 面 加 载 速度 。 针 对 这 种 情况 ， 
ASPNET 人 允许 开发 人 员 自 己 决定 是 否 使 用 ViewState， 以 避免 产生 不 必要 的 垃圾 数据 。 开 
发 人 员 可 以 在 页 面 的 <%@Page 中 启用 或 者 禁用 ViewState， 如 下 代码 所 示 。 

<!-- 下 面 这 行 代码 启用 了 ViewState--> 

<%@ Page Language="C#" RutoEventWireup="true"” CodeBehind="Pagel.aspx.cs" 

Inherits= "Test.Pagel" EnableViewState="true"%®> 


<!-- 下 面 这 行 代码 禁用 了 ViewState--> 

<%@ Page Language="C#" RutoEventWireup="true"” CodeBehind="Pagel.aspx.cs" 

Inherits= "Test.Pagel" EnableViewState="false"%®> 

如 果 一 个 控件 用 ViewState 来 进行 状态 保持 ， 而 控件 所 在 的 页 面 禁用 了 ViewState 被 ， 
那么 控件 的 状态 保持 机 制 就 不 能 正常 工作 。 这 时 候 ， 需 要 另外 一 种 状态 保持 机 制 一 一 控件 
状态 。 控 件 状态 的 工作 方式 与 视图 状态 类 似 ， 也 是 把 状态 数据 散 列 为 一 个 字符 串 ， 并 把 此 
字符 串 保 存 到 页 面 的 隐藏 域 中 。 与 视图 状态 不 同 的 是 ， 控 件 状 态 不 能 被 禁用 ， 始 终 有 效 。 
控件 状态 是 专 为 存储 控件 的 重要 数据 《〈 如 一 个 页 面 控件 的 页 数 ) 而 设计 的 ， 回 发 时 必须 用 
到 这 些 数 据 才 能 使 控件 正常 工作 〈 即 便 禁 用 视图 状态 也 不 受 影响 )。 

【 例 1-26】 控件 状态 。 

本 例 演示 控件 状态 在 自 定义 控件 中 的 使 用 ， 并 将 其 与 视图 状态 对 比 。 

(1) 创建 一 个 类 库 项 目 , 在 项 目 中 添加 一 个 继承 自 WebControl 的 类 HelloControlState。 
代码 如 下 : 


// 自 定义 控件 示例 ， 通 过 ControlState 保持 状态 
public class HelloControlState : WebControl 


(2) 在 HelloControlState 类 中 添加 3 个 属性 ， 注 意 与 例 1-25 区 别 ， 这 3 个 属性 并 不 保 


。S9 。 


第 1 篇 ASPNET 网 络 开发 关键 技术 


存在 ViewState 中 。 代 码 如 下 : 


#region 自 定义 属性 ， 通 过 ControlState 保存 这 些 属 性 
public string name { get; set; } 

public Color helloColor { get; set; } 
public Color nameColor { get; set; } 
#endregion 


(3) 在 HelloControlState 类 中 添加 一 个 内 部 类 HelloProperty， 这 个 类 用 于 向 控件 状态 
中 保存 和 恢复 控件 的 属性 。 代 码 如 下 : 


// 内 部 类 ， 用 于 封装 控件 的 属性 ， 以 便于 保存 和 读 取 

// 此 类 必须 声明 为 可 序列 化 

[Serializable] 

class HelloProperty 

{ 
public string name { get; set; } 
public Color helloColor { get; set; } 
public Color nameColor { get; set; } 

. 


名 提示: 要 保存 到 ControlState 中 的 属性 类 型 必须 是 可 序列 化 的 。.NET 中 的 简单 类 如 
string、int、double 都 是 可 序列 化 的 ， 如 果 是 自 定义 类 ， 默 认 情 况 下 都 是 不 可 序 
列 化 的 ， 可 以 在 类 前 面 加 上 [Serializable] 明 确 指 定 类 可 序列 化 。 


(4) 在 HelloControlState 类 中 重 写 SaveControlState0 方 法 ， 将 控件 属性 保存 到 
ControlState 中 。 代 码 如 下 : 


// 将 控件 属性 保存 到 Controlstate 
//<returns> 被 保存 的 Controlstate</returns> 
protected override object SaveControlState () 
{ 
HelloProperty property = new HelloProperty(); // 创 建 一 个 属性 类 
// 把 控件 属性 保存 到 属性 类 的 各 个 字段 中 
property.name = this.name; 
property.helloColor = this.helloColor; 
property.nameColor = this.nameColor; 
object o=base.SaveControlState();  // 调 用 基 类 的 SaveControlState() 方 法 
if (o != null) 
{ 
// 如 果 基 类 保存 的 Controlstate 不 为 空 
// 则 将 基 类 的 ControlState 与 该 控件 自 定义 的 属性 一 起 返回 
return new Pairl(o, property); 
} 
else 
{ 
// 如 果 基 类 保存 的 ControlState 为 空 ， 则 只 需要 返回 该 控件 自 定义 的 属性 


return property; 
} 
} 


(5) 在 HelloControlState 类 中 重 写 LoadControlState() 方 法 ， 从 保存 的 ControlState 中 恢 
复 控件 属性 。 代 码 如 下 : 


/// <summary> 


/// 从 ControlState 中 恢复 控件 属性 
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/// </summary> 
/// <param name="savedState"> 被 保存 的 ControlState</param> 
/// 这 个 方法 是 savecontrolState 方法 的 逆 过 程 。 读 取 数据 时 ， 应 按照 与 保存 数据 相 一 致 的 格式 


protected override void LoadControlState (object savedState) 

| 
if (savedState == null) return; // 没 有 被 保存 的 数据 则 返回 
// 如 果 被 保存 的 数据 为 Pair 类 ， 则 说 明 Pair 的 前 一 元 素 为 基 类 保存 的 数据 
// 后 一 元 素 为 当前 类 保存 的 数据 ( 即 HelloProperty) 


if (savedState is Pair) 


{ 
Pair pair = savedState as Pair; 
base.LoadControlState (pair.First); // 恢 复 基 类 保存 的 控件 状态 
HelloProperty property = pair.Second as HelloProperty; 
// 得 到 当前 类 保存 的 控件 状态 
if (property == null) return; 
/ /根据 读 取 的 控件 状态 数据 恢复 控件 属性 
this.name = property.name; 
this.helloColor = property.helloColor; 
this.nameColor = property.nameColor; 
} 
else if (savedState is HelloProperty) 
Ll 


// 如 果 被 保存 的 数据 是 HelloProperty 类 型 ， 则 直接 恢复 控件 属性 
HelloProperty property = savedState as HelloProperty; 
if (property == null) return; 
this.name = property.name; 
this.helloColor = property.helloColor; 
this.nameColor = property.nameColor; 

} 


else 


{ 
// 如 果 被 保存 的 数据 既 不 是 Pair， 也 不 是 HelloProperty， 那 么 调用 基 类 的 方法 恢复 控件 
态 


base.LoadControlState (SavedState) 


外 提示 : 如 果 自 定义 控件 用 控件 状态 保存 数据 ， 需 要 重 写 SaveControlState() 方 法 和 
LoadControlState() 方 法 。 这 两 个 方法 是 互 逆 的 过 程 。SaveControlState 按照 一 定 的 
格式 保存 数据 ， 而 LoadControlState 按照 相同 的 格式 把 保存 的 数据 读 出 。 


(6) 在 HelloControlState 类 中 重 写 OnInit0 方 法 ， 将 控件 注册 为 拥有 ControlState。 代 
码 如 下 : 

protected override void OnInit (EventArgs e) 
{ 

base.onInit (e); 

Page.RegisterRequiresControlState (this); 

// 将 此 控件 注册 为 具有 持久 性 控件 状态 

} 


全 提示 : 当 自 定义 控件 用 ControlState 保存 控件 数据 时 ， 必 须 把 控件 注册 为 具有 拥有 
ControlState。 只 有 经 过 注册 的 控件 ，ASP.NET 才 会 调用 其 LoadControlState0 和 
SaveControlState() 方 法 . 否则 , 即使 重 写 了 LoadControlState0 和 SaveControlState() 
方法 ， 这 些 方法 也 不 会 被 执行 。 
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(7) 在 HelloControlState 类 中 重 写 RenderContents() 方 法 ， 输 出 控件 内 容 。 代 码 如 下 : 


protected override void RenderContents (HtmlTextWriter writer) 


string cl = ColorTranslator.ToHtm]l (helloColor); 

// 将 .NET 的 颜色 转换 为 HTML 形式 
// 添 加 颜色 样式 ， 此 语句 将 生成 以 下 代码 : style="color: 颜 色 " 
writer.AddAttribute (HtmlTextWriterAttribute.Sstyle, "color:" + cl1); 
writer.RenderBeginTag (HtmlTextWriterTag.Span); 

// 输 出 一 个 span 开始 标记 <span> 

writer.Write ("你 好 "); // 输 出 你 好 
writer.RenderEndTag (); // 结 束 前 一 个 HTML 标记 ， 即 span 
// 以 下 语句 以 特定 颜色 在 span 中 输出 姓名 
string c2 = ColorTranslator.ToHtml (nameColor); 
writer.AddAttribute (HtmlTextWriterAttribute.Style, "color:" + c2); 
writer.RenderBeginTag (HtmlTextWriterTag.Span); 
writer.Write (name); 
writer.RenderEndTag (); 


(8) 新 建 一 个 ASPNET 项 目 ， 在 项 目 中 添加 一 个 页 面 ViewStateControlState.aspx 以 测 
试 HelloControlState 控件 。 在 页 面 上 放置 一 个 本 例 创建 的 HelloControlState 控件 和 一 个 例 
1-25 创建 的 HelloViewState 控件 ， 禁 用 页 面 的 ViewState。 页 面 代码 如 下 : 


<form id="forml" runat="server"> 
<div> 


下 面 这 个 控件 用 ViewState 保存 状态 <br /> 

<cc1:HelloViewState runat="server" ID="controll” name=" 初 始 值 " hello 
Color="Green" nameColor="Blue" /><br /> 

下 面 这 个 控件 用 ControlState 保存 状态 <br /> 

<ccl:HelloControlState runat="server" ID="control2" name=" 初 始 值 " hello 
Color="Green" nameColor="Blue" /><br /> 


输入 新 值 : <asp:TextBox ID="TextBoxl" runat="server"></asp:TextBox> 


<asp:Button ID="Button1l" runat="serVer" Text=" 设 置 新 值 " onclick= 
"Buttonl CLICK /> 
<asp:Button ID="Button2" runat="server" Text=" 回 发 " /> 


</div> 
</form> 


(9) 为 ViewStateControlState.aspx 页 面 上 的 Button 编写 事件 处 理 程序 ， 代 码 如 下 : 


protected void Buttonl Click(object 
sender, EventArgs e) 


{ 


(10) 运 行 ViewStateControlState.aspx 页 面 ， 
输入 新 值 以 进行 测试 。 可 以 看 到 ， 在 禁用 了 ”7 
ViewState 的 页 面 中 ， 保 存在 ViewState 中 的 属 和 下 | 
性 被 丢失 ,而 保存 在 ControlState 中 的 属性 工作 


controll .name 
control2.name 


TextBox] .Text; 


TextBox] .Text; 二 
netp://alecalhes~trelstate aspz| 工 属 


下 面 这 个 控件 用 WienState 保 存 杖 态 。 
你 好 初始 什 


下 面 这 个 控件 用 ControlState 保 存 状态 


EE 区 


正常 。 程 序 运行 结果 如 图 1.41 所 示 。 图 1.41 控件 状态 与 视图 状态 
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1.6.7 回 发 数据 和 事件 


ASPNET 的 页 面 呈现 为 HTML 时 会 生成 一 个 form， 页 面 回 发 时 会 把 form 及 form 上 
用 户 输入 的 数据 提交 给 服务 器 。ASPNET 标准 服务 器 端 控 件 可 以 处 理 这 些 回 发 的 数据 ， 而 
自 定义 控件 需要 编写 相应 代码 才能 完成 这 些 工作 。 

如 果 一 个 自 定 义 控 件 需 要 处 理 浏 览 器 的 回 发 数据 ， 则 需要 继承 IPostBackDataHandler 
接口 。IPostBackDataHandler 接口 以 下 有 两 个 方法 。 

口 LoadPostData 方法 : 此 方法 接收 从 浏览 器 回 发 的 数据 。 

口 RaisePostDataChangedEvent 方法 : 此 方法 产生 数据 更 改 事件 。 

【 例 1-27】 简单 的 TextBox 控件 。 

本 例 将 模仿 ASPNET 标准 控件 中 的 TextBox 编写 一 个 MyTextBox 控件 ， 这 个 控件 允 
许 用 户 在 其 中 输入 文本 , 并 可 以 处 理 回 发 的 这 些 文本 。 如果 新 输入 的 文本 与 原来 文本 不 同 ， 
则 触发 一 个 事件 。 

(1) 新 建 一 个 类 库 项 目 CustomControls。 

(2) 在 CustomControls 添加 一 个 MyTextBox 类 ， 该 类 从 WebControl 类 派生 ， 并 实现 
IPostBackDataHandler 接口 。 


public class MyTextBox:WebControl,IPostBackDataHandler 
{} 


(3) MyTextBox 控件 是 一 个 文本 控件 ， 控 件 标记 应 为 nput， 重 写 TagKey 属性 以 实现 
此 功能 。 代 码 如 下 : 

// 控 件 的 HTML 标记 为 input 

protected override HtmlTextWriterTag TagKey 

{ 


get 
{ 


} 


return HtmlTextWriterTag.Input; 
(4) 为 MyTextBox 控件 添加 一 个 text 属性 ， 该 属性 保存 在 ViewState 中 。 代 码 如 下 : 


private const string ViewStateKey="ViewStateText"; 
// 控 件 的 text 属性 

public string text 

{ 


get 
| 
if (ViewState [ViewStateKey] == null) 
return null; 
else 
return ViewState[ViewStateKey] as string; 
} 
set 


{ 
ViewState[ViewStateKey] = value; 


上 
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(5) 为 MyTextBox 控件 的 HTML 标记 添加 适当 的 属性 ， 代 码 如 下 : 
// 在 呈现 控件 前 为 控件 添加 属性 


protected override void AddAttributesToRender (HtmlTextWriter writer) 
| 
writer.AddAttribute (HtmlTextWriterAttribute.Type, "text"); 
// 添 加 属性 : type="text" 
// 添 加 属性 : style="border:solid 1px blue" 
writer.AddSstyleAttribute (HtmlTextWriterStyle.BorderSstyle, "solid" ); 
writer.AddSstyleAttribute (HtmlTextWriterStyle.BorderWidth, "lpx"); 
writer.AddStyleAttribute (HtmlTextWriterStyle.BorderColor, "blue"); 
// 添 加 属性 name=< 控 件 名 称 > 
writer.AddAttribute (HtmlTextWriterAttribute.Name, this.UniqueID); 
writer.AddAttribute (HtmlTextWriterAttribute.Value, text); 
// 添 加 属性 value=text 值 
base.AddAttributesToRender (writer); 
a 


(6) 将 MyTextBox 控件 注册 为 需要 处 理 回 发 ， 代 码 如 下 : 


protected override void OnInit (EventArgs e) 
{ 


base.OnInit(e) 7 


Page.RegisterRequiresPostBack (this); // 将 控件 注册 为 处 理 回 发 
} 


(7) 为 MyTextBox 控件 添加 一 个 textChanged 事件 : 


// 定 义 事件 myTextChanged， 当 文本 属性 时 触发 
public event EventHandler<EventArgs> myTextChanged; 


(8) 实现 IPostBackDataHandler 接口 的 LoadPostData 方法 以 处 理 回 发 数据 。 代 码 如 下 : 


// 处 理 回 发 数据 


public bool LoadPostData(string postDataKey, System.Collections. 


Specialized.NameValueCollection postCollection) 


{ 


string newText = postCollection[postDataKey]; // 取 得 控件 中 的 新 文本 


// 如 果 新 文本 与 旧 文 本 相同 ， 则 不 触发 Changed 事件 ， 否 则 触发 Changed 事件 
if (newText == text) 
return false; 
else 
{ 
text = newText; 
return true; 


} 
(9) 实现 IPostBackDataHandler 接口 的 RaisePostDataChanged() 方 法 以 触发 数据 更 改 寻 


代码 如 下 : 


// 触 发 数据 更 改 事件 

public void RaisePostDataChangedEvent () 

1 OnTextChanged (EventRArgs .Empty) 

人 virtual void OnTextChanged (EventArgs e) 
if (myTextChanged != null) 
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myTextChanged (this, e); 
} 


(10) 新 建 一 个 ASPNET 项 目 ， 在 项 目 中 添加 一 个 页 面 MyTextBoxPage.aspx 以 测试 
MyTextBox 控件 。 页 面 代 码 如 下 : 


<form id="forml" runat="server"> <div> 
<ccl:MyTextBox runat="server" ID=" 七 extbox1" 
onmytextchanged="textbox1 myTextChanged" /> 
<asp:Button ID="Button1"” runat="server" Text=" 回 发 " onclick="Buttonl 
Click™ /> 
DE 
<asp:Label ID="Labell" runat="server"” Text=" 输 出 结果 "></asp:Label> 
</div> 
</form> 


(11) 在 MyTextBoxPage.aspx 页 面 上 为 MyTextBox 控件 的 myTextChanged 事件 编写 如 
下 代码 : 


protected void textbox1 myTextChanged (object sender, EventArgs e) 
{ 

Label1 .Text = "myTextBox 文本 发 生 改变 ， 新 的 文本 为 "+textbox1l .text; 
jl: 


(12) 在 浏览 器 中 运行 MyTextBoxPage.aspx 页 面 ， 在 MyTextBox 控件 中 输入 文本 ， 单 
击 “ 回 发 ”按钮 ， 则 控件 中 的 文本 可 以 回 发 到 服务 器 ， 并 触发 myTextChanged 事件 。 运 行 
界面 如 图 1.42 所 示 。 


| 
文件 外。 锯 查 加 查看 WD 历史 G) 书签 @) 工具 中 帮助 如 


【< Ch -Ma 


1 js 人 参 


本 章 介绍 了 ASP.NET 编程 的 一 些 高 级 技术 。 首 先 分 析 了 ASPNET 页 面 生命 周期 和 事 
件 模型 ， 使 读者 从 一 定 深度 上 理解 ASP NET 技术 ， 接 着 介绍 了 母 版 、 主 题 、 用 户 控件 和 
自 定义 控件 ， 其 中 自 定义 控件 部 分 是 本 章 的 难点 。 
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ADONET 是 .NET 架构 中 用 于 访问 数据 库 的 组 件 ，ADONET 组 件 主要 存在 于 
System.Data 及 其 子 命令 空间 中 。 通 过 ADONET， 可 以 使 用 相同 的 方式 访问 和 操作 各 种 类 
型 的 数据 库 ， 如 Oracle、SQL Server 等 。 本 章 将 主要 以 访问 SQL Server 数据 库 为 例 说 明 
ADO.NET 的 功能 和 使 用 方法 。 


2.1 ADO.NET 概述 


ADO.NET 通过 数据 处 理 将 数据 访问 分 解 为 多 个 可 以 单独 使 用 或 一 前 一 后 使 用 的 不 连 
续 组 件 。ADO.NET 包含 用 于 连接 到 数据 库 、 执 行 命令 和 检索 结果 的 .NET Framework 数据 
提供 程序 。ADO.NET 用 于 访问 和 操作 数据 的 两 个 主要 组 件 是 .NET Framework 数据 提供 程 
序 和 DataSet， 如 图 2.1 所 示 。 


.NET Framework 数 据 提供 程序 
Connection DataAdapter 


SelectCommand | DataTable 


图” DataRowCollection 
Command | 

| DataColumnCollection 

ConstraintCollection 
DataReader 号 


DataRelationCollection 
t 4 
EE ' 
XML 
数据 库 


图 2.1 ADONET 结构 


.NET Framework 数据 提供 程序 是 专门 为 数据 操作 及 快速 、 只 进 、 只 读 访问 数据 而 设计 
的 组 件 。Connection 对 象 提供 到 数据 源 的 连接 。 使 用 Command 对 象 可 以 访问 用 于 返回 数 
据 、 修 改 数据 、 运 行 存 储 过 程 ， 以 及 发 送 或 检索 参数 信息 的 数据 库 命 令 。DataReader 可 从 
数据 源 提供 高 性 能 的 数据 流 。DataAdapter 在 DataSet 对 象 和 数据 源 之 间 起 到 桥梁 作用 。 
DataAdapter 使 用 Command 对 象 在 数据 源 中 执行 SQL 命令 以 向 DataSet 中 加 载 数据 ， 并 
将 对 DataSet 中 数据 的 更 改 协调 回 数据 源 。 

ADONET DataSet 是 专门 为 独立 于 任何 数据 源 的 数据 访问 而 设计 的 ， 可 以 用 于 多 种 不 
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同 的 数据 源 ， 用 于 XML 数据 ， 或 用 于 管理 应 用 程序 本 地 的 数据 。DataSet 包含 一 个 或 多 个 
DataTable 对 象 的 集合 ， 这 些 对 象 由 数据 行 和 数据 列 及 有 关 DataTable 对 象 中 数据 的 主键 、 
外 键 、 约 束 和 关系 信息 组 成 。 


2.2 连接 数据 库 


数据 库 与 应 用 程序 (如 ASPNET 应 用 程序 ) 是 服务 器 一 一 客户 端的 关系 ， 应 用 程序 使 
用 数据 库 提供 的 服务 完成 数据 存储 、 修 改 、 查 询 等 功能 。 应 用 程序 使 用 数据 库 的 第 一 步 是 
连接 到 数据 库 ， 之 后 才能 进行 其 他 操作 。 


2.2.1 ”数据 库 连接 类 DbConnection 


.NET Framework 中 的 System.Data.Common.DbConnection 类 是 一 个 抽象 类 ， 表 示 到 数 
据 库 的 连接 。 从 DbConnection 类 派生 出 一 组 具体 的 数据 库 连 接 类 ， 分 别 表示 到 一 种 特定 数 
据 源 的 连接 ， 如 图 2.2 所 示 。 


DbConnection 


OracleConnection [SqlConnection 


[OdbeConnection | OleDbConnection 


图 2.2 DbConnection 类 及 其 派生 类 


图 2.1 中 ，OdbcConnection 类 表示 到 ODBC 数据 源 的 连接 ，OleDbConnection 类 表示 
到 OleDB 数据 源 的 连接 ，OracleConnection 类 表示 到 Oracle 数据 库 的 连接 ，SqlConnection 
类 表示 到 MS SQL Server 数据 库 的 连接 。DbConnection 类 的 各 个 派生 类 的 属性 方法 大 致 相 
同 ， 下 面 以 SqlConnection 为 例 来 说 明 连 接 到 数据 库 的 方法 。 
SqlConnection 类 的 主要 属性 见 表 2-1。 
说 明 


表 2-1 ee 
怕 性 
用 于 连接 到 SQL Server 数据 库 的 连接 字符 串 


类 型 i 
ConnectionString 于 连接 
Database 连接 到 的 数据 库 的 名 字 
DataSource SQL Server 数据 源 名 称 


State i 连接 的 当前 状态 
ConnectionTimeout [it | | RW | | ”RW | 连接 超 时 值 ( 以 秒 为 单位 》 


二 


根据 身份 验证 方式 的 不 同 ， 用 于 连接 SQL Server 的 连接 字符 串 也 有 所 不 同 。 如 果 使 
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Windows 身份 验证 ， 则 连接 字符 串 具 有 如 下 形式 。 
Data Source= Initial Catalog= 数 据 库 名 ;Integrated Security=True 
如 果 使 用 SQL Server 身份 验证 ， 则 连接 字符 串 具 有 如 下 形式 。 
Server=SQLServer 实例 名 称 ; Database= 数 据 库 名 ; user id= 用 户 名 ;password= 密 码 


SqlConnection 类 的 State 属性 表示 当前 数据 库 连 接 状 态 ， 其 值 是 一 个 ConnectionState 
枚 举 类 型 。ConnectionState 枚 举 类 型 共 包含 6 个 值 ， 表 示 6 种 数据 库 连 接 状态 。 


口 Closed: 连接 处 于 关闭 状态 。 
口 Connecting: 正在 与 数据 源 连接 。 
口 Open: 连接 处 于 打开 状态 。 
口 Executing: 正在 执行 命令 。 
口 Fetching: 正在 检索 数据 。 
口 Broken: 连接 中 断 。 
SqlConnection 类 的 主要 方法 如 下 : 
public SqlConnection () // 构 造 函 数 
public SqlConnection(string connectinstring); // 构 造 函 数 
public override void Open(); // 打 开 连 接 
public override void Close(); // 关 闭 连 接 
public override void ChangeDatabase (string database); 
// 为 打开 的 连接 更 改 数据 库 
public SqlCommand CreateCommand(); // 创 建 一 个 SQL 命令 


2.2.2 连接 到 SQL Server 


使 用 SqlConnection 连接 到 SQL Server 数据 库 的 方法 就 是 先 用 连接 字符 串 创 建 一 个 
SqlConnection 对 象 ， 然 后 调用 SqlConnection 类 的 Connect(0 方 法 即 可 。 连 接 字符 串通 常 比 
较 长 ， 如 果 记 不 住 ， 在 Visual Studio 中 可 以 自动 生成 连接 字符 串 ， 操 作 步 骤 如 下 。 

(1) 打开 任意 一 个 ASPNET 页 面 。 

(2) 从 工具 箱 的 “数据 ”选项 卡 中 选择 SqlDataSource 控件 拖 放 到 页 面 上 ， 单 击 控件 
的 智能 标记 〔〈 控 件 右上 角 的 小 三 角 )， 则 弹出 智能 标记 面板 ， 如 图 2.3 所 示 。 


ET ET EET 


图 2.3 SqlDataSource 控件 的 智能 标记 面板 


(3) 在 图 2.3 所 示 的 智能 标记 面板 中 单 击 “ 配 置 数 据 源 ...” 按 钮 ， 则 弹出 “配置 数据 
源 ” 向 导 对 话 框 ， 如 图 2.4 所 示 。 

(4) 单 击 “新 建 连接 ”按钮 ， 则 弹出 “添加 连接 ”对 话 框 ， 如 图 2.5 所 示 。 

(5) 在 “添加 连接 ”对 话 框 中 设置 所 要 连接 的 数据 库 信息 ， 然 后 单 击 “ 测 试 连接 ” 按 
钮 。 提 示 连 接 成 功 后 ， 单 击 “高 级 ”按钮 ， 就 会 弹出 “高 级 属性 ”对 话 框 ， 如 图 2.6 所 示 。 
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在 图 2.6 对 话 框 的 底部 ， 就 显示 了 当前 数据 库 连 接 所 对 应 的 连接 字符 串 。 把 这 个 字符 串 复 
角 


出 来 ， 就 可 以 在 程序 中 使 


用 。 


EE Jo 
人 
数据 源 G) 
icrosoft SQL Server (SqlClient) 更 改 加 ). 
ME) 
ER 
中 sam | 合用 Windovs 身份 对 证 四 
恒 | 全 SqL Server 身价 证 加 
应 用 各 床 连 接 歼 所 库 应 使 用 旦 个 到 连接 (7)? 二 
El | Ee 
| | 厂 保 方 志 码 他) 
1 
IE 
| FT 
| | 
区 加 加 tU) 
万 时 矣 三 天 于 元 成 如 取 清 
加 
图 2.4 “配置 数据 源 ” 对 话 框 图 2.5 “添加 连接 ”对 话 框 


国际 


Type Systen Versio Latest 
日 来 源 
AttachDbFilenane 
Context Connection False 
Qocal) 
Failover Partner 
Initial Catalog test 
Vser Instance False 
日 上 下 文 


Application Wane .Net SqlClient Data 


Data Source 


ets Source= Oocal);Initial Catalog=test;Int 


| 确定 | 了 


图 2.6 连接 高 级 属性 对 话 框 
数据 库 连 接 是 一 种 宝贵 资源 ， 在 使 用 完毕 连接 以 后 ， 要 及 时 调用 Close0 方 法 关闭 数据 
库 连 接 。 下 面 的 代码 演示 了 这 一 过 程 。 

// 连 接 字符 串 


string connstring "Data Source=(local);Initial Catalog=Northwind; 
Integrated Security=True"; 


SqlConnection conn = new SqlConnection(connstring); 
conn.Open(); 
// 在 这 里 使 用 数据 库 连 接 


conn.Close(); 


// 创 建新 的 连接 对 象 
// 打 开 连 接 


// 使 用 完毕 后 关闭 连接 
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为 了 确保 数据 库 连 接 能 够 总 是 被 关闭 ， 而 不 受 代码 中 可 能 出 现 的 异常 影响 ， 通 常 把 关 
闭 数据 库 连 接 的 代码 放 到 finally 块 中 ， 代 码 如 下 : 


string connstring="server=(local) ;database=Northwind;user 1d=sa> 
password="; 
SqlConnection conn = new SqlConnection(connSstring); 
try 
{ 
conn.Open (); 
// 在 这 里 使 用 数据 库 连 接 
} 
finally 
{ 
conn.Dispose (); 


} 

上 述 代 码 还 有 一 种 更 简洁 的 书写 方式 ， 就 是 使 用 using 语句 ， 代 码 如 下 : 

string connString = "server=(local) ;database=Northwind;user id=sa;password 
=Sa"7 


using (SqlConnection conn = new SqlConnection (connString) ) 


{ 


conn.Open(); 
// 使 用 数据 库 连 接 conn 
y 


使 用 using 语句 时 ， 不 论 在 using 块 中 是 否 发 生 异 常 ， 在 退出 using 块 时 都 会 调用 对 象 
的 Dispose() 方 法 释放 被 使 用 的 对 象 ， 在 这 里 就 是 数据 库 连 接 。 


名 注意 : 在 using 关键 字 后 面 括号 里 的 对 象 必 须 实现 IDisposable 接口 。 


为 了 便于 程序 移植 ， 通 常 把 连接 字符 串 写 到 配置 文件 web.config 中 ， 而 不 是 硬 编码 到 
C# 代 码 中 。 这 样 ， 当 需要 连接 的 数据 源 发 生变 化 时 《〈 例 如 把 程序 从 开发 环境 部 署 到 用 户 环 
境 )， 就 可 以 通过 修改 配置 文件 实现 改变 数据 库 连接 ， 而 不 需要 修改 C# 代 码 和 重新 编译 。 
把 连接 字符 串 保 存 到 配置 文件 时 ， 要 保存 在 ConnectionStrings 结 点 中 ， 代 码 如 下 : 

<connectionStrings> 

<add name="database" connectionString="Data Source=.; ， DataBase= 

Northwind; Integrated Security =True; " providerName="System.Data. 

SqlCclient" /> 

</connectionstrings> 
于 配置 文件 中 可 以 保存 多 个 连接 字符 串 ， 为 了 区 别 ， 每 个 连接 字符 串 都 有 一 个 唯一 
的 名 字 ， 在 配置 文件 中 用 name 属性 标识 。 程 序 中 可 以 根据 名 字 找 到 某 一 个 特定 的 连接 字 
符 串 。 在 程序 中 获取 连接 字符 串 需 要 使 用 ConfigurationManager 类 。ConfigurationManager 
类 是 专门 用 于 访问 配置 文件 的 一 个 类 ， 获 取 连 接 字符 串 的 代码 如 下 : 

string connString = ConfigurationManager.ConnectionSstrings["database"]. 

Connectionstring; 


【 例 2-1】 连接 到 SQL Server。 
本 例 演示 如 何 用 代码 连接 到 SQL Server 数据 库 。 本 例 使 用 的 数据 库 中 SQL Server 速成 
版 SQL Server Express)。 如 果 要 连接 到 其 他 版 本 ， 仅 需要 修改 代码 中 的 连接 字符 串 即 可 。 
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全 提示 : 使 用 SQL Server 速成 版 的 一 个 好 处 是 可 以 直接 在 Visual Studio 中 添加 数据 库 ， 从 
而 数据 库 文件 与 项 目 中 的 其 他 文件 在 同一 路 径 下 ， 便 于 将 来 程序 的 部 署 。 另 外 一 
个 好 处 是 程序 运行 不 需要 单独 安装 SQL Server，.NET Framework 中 自 带 了 SQL 
Server 速 成 版 .如 果 用 SQL Server 标准 版 或 者 企业 版 ,就 需要 单独 安装 SQL Server 
环境 ， 还 需要 附加 数据 库 等 一 些 额 外 操作 。 


(1) 创建 一 个 ASPNET Web 应 用 程序 ADONET。 

(2) 在 项 目 上 右 击 ， 从 弹出 的 快捷 菜单 中 选择 “添加 新 项 ”选项 。 在 “添加 新 项 ”对 
话 杠 中， 选择 “SQL Server 数据 库 ”， 单 击 “ 确 定 ”按钮 。 这 样 就 在 项 目 中 添加 了 一 个 SQL 
Server 数据 库 。 

(3) 打开 配置 文件 web.config， 在 其 中 添加 连接 字符 串 ， 代 码 如 下 。 


<connectionStrings> 
<add name="database" connectionString="Data Source=.\SQLEXPRESS; 
AttachDbFilename= |DataDirectory|\Databasel .mdf;Integrated Security= 
True;User Instance=True" 
providerName="System.Data.SqlClient" /> 
</connectionstrings> 


(4) 在 项 目 中 添加 一 个 页 面 ConnectionSample.aspx， 在 页 面 上 放置 一 个 Button 和 一 个 
Label。 代 码 如 下 : 


<form id="forml" runat="server"> 
<div> 
<asp:Button ID="Buttonl" runat="server" Text=" 连 接 " onclick="Buttonl 
Click” /> <br /> 
<asp:Label ID="Labell" runat="server"” Text=" 输 出 信息 "></asp:Label> 
</div> 
</form> 


(5) 为 Button 的 Click 事件 编码 ， 连 接 到 数据 库 ， 然 后 关闭 连接 。 代 码 如 下 : 


protected void Button1l Click(object sender, EventArgs e) 
a 
// 连 接 字符 串 
string connString = ConfigurationManager.ConnectionStrings ["database"] . 
Connectionstring; 
Labell.Text = ""; 
using (SqlConnection conn = new SqlConnection (connString) ) 


{ 


conn.Open (); 
// 打 开 连 接 文件 下， 编辑 下 ) 查看 WD 历史 (GS) 书签 @) 
Label1.Text += "连接 已 经 打开 。<br/>"; 外 - cx - 
conn-Close (); Google [sp ne. Fiet 5] SI- 
// 关 闭 连接 口 er - 
Label1 .Text += "连接 已 经 关闭 。<br/>"; EE 
} 大 到 | 

连接 已 经 打开 。 
连接 已 经 关闭 。 辟 
有 成 EJEI 


(6) 运行 页 面 ， 单 击 “ 连 接 ” 按 钮 ， 运 行 结果 如 图 2.7 


图 2.7 连接 数据 库 示 例 
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2.3 修改 数据 


应 用 程序 对 数据 库 的 操作 总 体 可 以 分 为 两 类 : 修改 数据 和 读 取 数据 。 修 改 数据 包含 增 
加 、 删 除 、 更 新 数据 。 本 节 将 介绍 ADO.NET 中 修改 数据 的 相关 类 和 方法 。 


2.3.1 数据 库 命 令 类 DbCommand 


System.Data.Common.DbCommand 类 表示 要 对 数据 库 执行 的 SQL 语句 或 存储 过 程 。 
DbCommand 类 是 一 个 抽象 类 ，.NET Framework 为 不 同 的 数据 库 管理 系统 分 别 创建 了 从 
DBCommand 类 派生 的 具体 的 数据 库 命 令 类 ， 如 图 2.8 所 示 。 


OdbcCommand OleDbCommand OracleCommand SqlCommand 


图 2.8 DbCommand 类 及 其 派生 类 


图 2.7 中 ,OdbcCommand 类 表示 ODBC 数据 源 的 命令 ,OleDbConnection 类 表示 OleDB 
数据 源 的 命令 ，OracleConnection 类 表示 Oracle 数据 库 的 命令 ，SqlConnection 类 表示 MS 
SQL Server 数据 库 的 命令 。 这 一 组 数据 库 命 令 类 的 使 用 大 致 相似 ， 下 面 以 SqlCommand 为 
例 介 绍 。 

SqlCommand 类 的 主要 属性 见 表 2-2。 


表 2-2 SqlCommand 类 主要 属性 


属 性 说 了 明 


CommandType | commandType 命令 类 型 ， 如 存储 过 程 、SQL 语句 等 
CommandText | string 命令 文本 

Connection | SqlConnection 此 命令 所 使 用 的 SqlConnection 对 象 
Parameters | SqlParameterCollection R 命令 的 参数 


以 秒 为 单位 的 命令 超时 时 间 
命令 所 属 的 事务 对 象 


CommandTimeout int RW 
SqlCommand 类 的 CommandText 属性 是 一 个 字符 串 ， 表示 命令 的 文本 ， 其 值 可 以 是 一 

组 SQL 语句 ， 如 select * from Customers， 也 可 以 是 一 个 存储 过 程 ， 如 SalesByCategory。 
SqlCommand 类 的 CommandType 属性 是 一 个 CommandType 枚 举 类 型 的 值 ， 表 示 

SqlCommand 对 象 的 CommandText 是 什么 含义 。CommandType 枚 举 包 含 以 下 值 。 


Transaction 


yy 
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口 Text: CommandText 中 包含 SQL 命令 文本 。 

口 StoredProcedure: CommandText 中 包含 存储 过 程 名 称 。 

口 TableDirect: CommandText 中 包含 数据 库 表 的 名 称 。 

了 Parameters 和 Transaction 属性 将 在 本 章 后 续 内 容 中 详细 介绍 。 

SqlCommand 类 主要 方法 如 下 。 

(1) 构造 函数 

SqlCommand 类 共有 4 个 构造 函数 ， 代 码 如 下 : 

public SqlCommand () :// 默 认 构造 函数 

public SqlCommand (string commandText) ; // 用 指定 的 SQL 命令 文本 初始 化 一 个 命令 


public SqlCommand (string commandText， SqlConnection connection) 
// 用 指定 的 SQL 命令 文本 和 数据 库 连接 对 象 初始 化 一 个 命令 
// 用 指定 的 SQL 命令 文本 、 数 据 库 连 接 和 事务 对 象 初始 化 一 个 命令 


public SqlCommand (string commandText, SqlConnection connection, 
SqlTransaction transaction); 


(2) 执行 命令 
SqlCommand 类 可 以 以 几 种 不 同 的 方式 执行 ， 代 码 如 下 : 


public override int ExecuteNonQuery(); // 执 行 SQL 命令 并 返回 受 影响 的 行 数 

public override object ExecuteScalar(); // 执 行 SQL 命令 ， 并 返回 结果 集中 第 一 
// 行 第 一 列 的 值 

public SqlDataReader ExecuteReader() ; // 执 行 SQL 命令 , 并 返回 一 个 SqlData- 
//Reader 对 象 


public SqlDataReader ExecuteReader (CommandBehavior behavior); 


如 果 要 执行 的 SQL 命令 没有 返回 结果 ， 如 INSERT、DELETE、CREATE TABLE 等 全 
令 ， 那 么 应 该 调用 ExecuteNonQuery(0) 方 法 。 如 果 要 执行 的 命令 仅 返回 一 个 值 ， 如 SELECT 
COUNT(*) 命 令 等 , 那么 应 该 调用 ExecuteScalar0 方 法 。 如 果 要 执行 的 命令 会 返回 多 行 多 列 ， 
则 应 该 使 用 ExecuteReader() 方 法 。ExecuteReader() 方 法 返回 一 个 SqlDataReader 对 象 ， 关 于 
SqlDataReader， 将 在 后 续 内 容 中 介绍 。 


2.3.2 ”命令 参数 DbParameter 


执行 SQL 命令 时 ， 大 多 数 情况 下 都 需要 向 SQL 语句 或 者 存储 过 程 传递 参数 。 在 .NET 
Framework 中 , 使 用 System.Data.Common.DbParameter 类 表示 SQL 命令 的 参数 .DbParameter 
是 一 个 抽象 类 。 如 同 DbConnection 和 DbCommand 类 一 样 ， 对 于 不 同 的 数据 库 管 理 系 
统 ，.NET Framework 中 有 不 同 的 DbParameter 类 的 派生 类 ， 包 括 OdbcParameter、 
OleDbParameter、OracleParameter 和 SqlParameter。SqlParameter 类 的 主要 属性 见 表 2-3 。 


表 2-3 SqlParameter 类 主要 属性 


属 性 类 型 读 说 上 明 
ParameterName 1 RW 参数 名 称 
DbType SqlDbType RW 参数 的 SQL 数据 类 型 
Direction 参数 传递 方向 输入 输出 ) 
IsNullable bool RW 参数 是 否 可 以 为 空 
Value object RW 参数 的 值 


wk 
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SqlParameter 类 的 主要 构造 函数 如 下 : 


public SqlParameter () ; // 默 认 构 造 函数 
public SqlParameter (string paramName, object value); 


// 用 参数 名 称 和 值 初始 化 一 个 新 的 SqlParameter 对 象 
public SqlParameter (string paramName, SqlDbType dbType); 

// 创 建 一 个 具有 指定 名 称 和 类 型 的 SqlParameter 对 象 
public SqlParameter (string paramName, SqlDbType dbType, int size); 


// 创 建 一 个 具有 指定 名 称 、 类 型 、 大 小 的 SqlParameter 对 象 
要 执行 带 参数 的 SQL 语句 或 者 存储 过 程 时 ， 必 须 向 SqlCommand 中 添加 所 需 的 参数 。 


方法 是 创建 SqlParameter 类 的 实例 ， 并 调用 SqlCommand.Parameters.Add() 方 法 将 参数 添加 


到 命 


令 中 。 


2.3.3 修改 数据 


对 数据 库 数据 的 修改 主要 包括 插入 (INSERT)、 删 除 (DELETE)、 更 新 (UPDATE) 


3 种 操作 ,这 3 种 操作 通常 分 别 对 应 于 一 条 SQL 语句 。 如 前 所 述 , ADO.NET 的 DbCommand 
类 表示 SQL 命令 ， 可 以 用 DbCommand 执行 各 种 数据 修改 。 


用 ADONET 执行 一 条 SQL 命令 通常 包含 以 下 几 个 步骤 。 

(1) 获取 连接 字符 串 ， 创 建 到 数据 库 的 连接 。 

(2) 打开 连接 。 

(3) 创建 命令 对 象 。 

(4) 为 命令 添加 参数 。 

(5) 执行 命令 ， 获 得 命令 结果 。 

(6) 关闭 命令 。 

(7) 关闭 连接 。 

(8) 处 理 命令 结果 。 

【 例 2-2】 用 户 注册 。 

本 例 以 用 户 注 册页 面 为 例 ， 演 示 如 何 用 ADO.NET 向 数据 库 插入 数据 。 
(1) 创建 一 个 ASPNET 应 用 程序 。 

(2) 在 项 目 中 添加 一 个 数据 库 ， 在 数据 库 中 添加 一 个 表 LoginUser， 为 表 添 加 三 列 ， 


各 列 的 名 称 、 字 段 类 型 如 下 : 


口 UserID: 登录 用 户 ID，nvarchar(10) 类 型 ， 不 可 为 空 ， 主 键 。 

口 UserName: 用 户 名 称 ，nvarchar(20) 类 型 ， 不 可 为 空 。 

口 Password: 登录 密码 ，nvarchar(20) 类 型 ， 不 可 为 空 。 

(3) 在 项 目 中 添加 一 个 注册 页 面 RegisterPage.aspx， 在 页 面 中 放置 3 个 TextBox， 用 于 


输入 注册 信息 ， 放 置 一 个 Button， 以 完成 注册 功能 ， 页 面 代码 如 下 : 


<form id="forml" runat="server"> 
<div> 
用 户 登 录 ID: <asp:TextBox ID="TextBoxl" runat="server"></asp:TextBox> 
<br /> 
用 户 名 称 : <asp:TextBox ID="TextBox2" runat="server"></asp:TextBox><br /> 
登录 密码 : <asp:TextBox ID="TextBox3" runat="server" TextMode="Password"> 
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</asp:TextBox> <br /> 


<asp:Button ID="Buttonl" runat="server"” Text=" 注 册 " /><br /> 
</div> 


</form> 


(4) 在 “注册 ”按钮 的 Click 事件 中 编写 代码 ， 完 成 用 户 注册 功能 。 代 码 如 下 : 


protected void Button1l Click(object sender, EventArgs e) 


t 


} 


// 从 配置 文件 中 得 到 连接 字符 串 

string E; = ConfigurationManager.ConnectionSstrings["database"]. 
ConnectionSstring; 

SqlConnection connection = new SqlConnection(s); // 创 建 连接 对 象 


// 构 建 插入 SQL 语句 
string sql = "insert into loginuser (userid,username,password)" 
+" values (@id, @name, @pass)"; 

SqlCommand command = new SqlCommand(sql,connection);  // 创 建 命令 对 象 
SqlParameter pl = new SqlParameter ("@id",TextBox].Text); 

// 创 建 一 个 命令 参数 userid 
command.Parameters.Add (p1); // 把 命令 参数 附加 到 命令 
// 构 建 和 附加 其 他 两 个 参数 
SqlParameter p2 = new SqlParameter("@name", TextBox2.Text); 
command.Parameters.Add (p2); 

SqlParameter p3 = new SqlParameter("@pass", TextBox3.Text); 
command.Parameters .Add (p3); 
string message=""; 


作 到 天 

! connection.Open (); // 打 开 连 接 
int n = command.ExecuteNonQuery (); // 执 行 命令 ， 得 到 受 影响 的 行 数 
age = “注册 成 功 。 用户 信息 已 经 保存 在 数据 库 。"; 

, A 三 "数据 未 能 保存 。"; 


catch (Exception ex) 


// 要 注意 处 理 错误 提示 信息 中 的 换行 符 
message = "保存 用 户 信息 过 程 中 出 错 。" 
+ ex.Message.Replace("\r", "\\r ") .Replace("\n","\\n"); 
} 
finally 
command.Dispose(); 
connection.Close(); 


| 

// 向 客户 端 返回 JavaScript， 提 示 执 行 结果 

Page.ClientScript.RegisterStartupScript (this.GetType(), 
"register", "<script>alert(\""+message+"\");</script>"); 


(5) 运行 RegisterPage.aspx 页 面 ， 测 试 注册 功能 。 运 行 界面 如 图 2.9 所 示 。 

【 例 2-3】 修改 密码 。 

本 例 以 修改 用 户 密码 页 面 为 例 ， 说 明 如 何 用 ADO.NET 更 新 数据 库 数据 。 

(1) 打开 例 2-2 所 创建 的 ASPNET 项 目 。 

(2) 在 项 目 中 添加 一 个 页 面 ChangePass.aspx, 在 页 面 上 放置 相应 控件 以 进行 密码 修改 ， 
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页 面 代码 如 下 : 


广 册页 面 ( 鞭 入 数据 示例 》 -Worills REGEE 


< 加 和 用 记忆 人 末了 


Google | 

注册 1 
用 户 登 录 ID，[aserod 

用 户 名 称 ， bill gates 

登录 密码 ， 

| 


| 正在 从 1ocalhost 传送 数据 … 


图 2.9 用 户 注册 页 面 


<form id="forml" runat="server"> 

<div> 

用 户 ID: <asp:TextBox ID="TextBoxl" runat="server"></asp:TextBox><br /> 
原 密码 : 

<asp:TextBox ID="TextBox2" runat="server" TextMode="Password" ></asp: 
TextBox><br /> 

新 密码 : 

<asp:TextBox ID="TextBox3" runat="server" TextMode="Password"></asp:Text 
Box><br /> 

<asp:Button ID="Buttonl"” runat="server"” Text=" 修 改 密码 " onclick="Buttonl 
Click" /><br /> 

</div> 

</form> 


(3) 为 ChnagePass.aspx 页 面 的 “修改 密码 ”按钮 编写 代码 ， 完 成 密码 修改 功能 。 代 码 
如 下 : 


protected void Buttonl] Click(object sender, EventArgs e) 
{ 


// 从 配置 文件 中 得 到 连接 字符 串 

string s = ConfigurationManager.ConnectionStrings ["database"] . 
Connectionstring; 

SqlConnection connection = new SqlConnection(s);  ”// 创 建 连接 对 象 
// 构 建 更 新 用 的 SQL 语句 

string sql = "update loginuser set password = @newpass " 


+"where userid = Q@id and password = @oldpass"; 
SqlCommand command = new SqlCommand(sql, connection); // 创 建 命令 对 象 
SqlParameter pl = new SqlParameter("@id", TextBox].Text); 

// 创 建 一 个 命令 参数 userid 

command.Parameters.Add (p1); // 把 命令 参数 附加 到 命令 
/ /构建 和 附加 其 他 两 个 参数 
SqlParameter p2 = new SqlParameter("@oldpass", TextBox2.Text); 
command.Parameters.Add (p2); 
SqlParameter p3 = new SqlParameter("@newpass", TextBox3.Text); 
command.Parameters.Add (p3); 
string message = ""; 


try 

{ 
connection.Open(); // 打 开 连 接 
int n = command.ExecuteNonQuery (); // 执 行 命令 ， 得 到 受 影响 的 行 数 
if (n == 1) 
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message = "密码 修改 成 功 。"; 


else 


message = "不 存在 此 用 户 或 者 原始 密码 不 对 。"; 


catch (Exception ex) 


{ 


} 


// 要 注意 处 理 错误 提示 信息 中 的 换行 符 
message = "更 改 密码 过 程 中 出 错 。" 


+ ex.Message.Replace("\r", "\\r ") -Replace("\n"，"\N\n") 7 


finally 


{ 


command.Dispose(); 
connection.Close(); 


} 
// 向 客户 端 返回 JavaScript， 提 示 执 行 结果 
Page.ClientScript.RegisterStartupScript (this.GetType(), 


ph 


"register", "<script>alert(\"" + message + "\");</script>"); 


(4) 运行 ChangePass.aspx 页 面 ， 运 行 结果 如 图 2.10 所 示 。 


修改 容 码 -Worilla ire 


| 正在 从 localhost 传送 数据 … 


图 2.10 修改 密码 页 面 


24 查询 数据 


日 于 SQL 查询 操作 的 复杂 性 , 使 用 ADO.NET 从 数据 库 查询 数据 要 比 修改 数据 更 加 复 


杂 。 对 数据 库 的 查询 要 用 到 ADO.NET 中 的 许多 类 ,除了 在 修改 数据 时 用 到 的 DbCommand、 
DbConnection、DbParameter 类 以 外 ， 还 要 到 DataReader、DataSet、DataAdapter 等 类 。 


2.4. 


返回 


对 数据 库 进行 查询 时 ， 大 多 数 查 询 可 能 返回 多 个 数据 ， 例 如 查询 满足 一 定 条 件 的 学 4 
信息 姓名、 学 号 、 班 级 等 )， 而 有 的 查询 只 会 返回 一 个 数据 ， 例 如， 查询 满足 条 件 的 学 和 
数量 (会 返 


口 


单个 值 


1 查询 单个 值 


We 


一 个 整数 )。 这 两 种 不 同 的 查询 方式 ， 在 ADO.NET 中 用 不 同 的 语句 来 完成 ， 


的 查询 语句 用 DbCommand.ExecuteScalar 语法 执行 ， 返 回 多 个 值 的 查询 语句 操 


作 较 为 复杂 ， 会 在 本 章 后 续 内 容 中 加 以 介绍 。 
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返回 单个 值 的 查询 语句 通常 都 是 聚合 函数 ， 如 COUNT、MAX、AVERAGE 等 ， 但 某 
些 时 候 非 聚合 函数 也 会 返回 单个 值 ， 例 如 以 主键 为 条 件 进行 查询 单个 列 时 。 以 SQL Server 
提供 的 Northwind 示例 数据 库 为 例 ， 下 列 SQL 查询 语句 都 是 返回 单个 值 的 查询 语句 。 

select count (*) from customers where Country='"UK" 

/* 查 询 来 自 UK 的 顾客 总 数 */ 

select sum(UnitPrice*Quantity) from [Order Details] where OrderID="'10248"' 

/* 查 询 订单 的 总 金额 */ 

select Phone from Customers where CustomerID="'ALFKI' 


/* 查 询 ALFKI 顾客 的 联系 电话 */ 


【 例 2-4】 用 户 信息 统计 。 

本 例 将 统计 注册 用 户 汇总 信息 ， 注 册 用 户 一 共 多 少 人 ， 男 女 各 多 少 人 。 

(1) 打开 例 2-2 所 创建 的 ASPNET 项 目 。 

(2) 打开 项 目 中 的 数据 库 ， 为 LoginUser 表 添 加 两 个 字段 。 

口 RegisterDate: 用 户 注册 日 期 ，datetime 类 型 ， 可 空 。 

口 Sex: 性 别 ，nchar(1) 类 型 ， 可 空 ，M 为 男 ，F 为 女 。 

(3) 在 项 目 中 添加 一 个 页 面 UserStatistics.aspx， 在 页 面 上 放置 相应 控件 以 统计 用 户 数 
并 显示 结果 ， 页 面 代码 如 下 : 

<form id="forml" runat="server"> 

<div> 


<asp:Button ID="Buttonl" runat="server" Text=" 统 计 用 户 数 " onclick= 
"Buttonl1 Click" /><br /> 

总 用 户 数 : 

<asp:Label ID="total" runat="server" Text="Label"></asp:Label><br /> 
其 中 男 用 户 

<asp:Label ID="male" runat="server" Text="Label"></asp:Label> 人 ， 

女 用 户 <asp:Label ID="female" runat="server" Text="Label"></asp:Label> 人 ， 
未 填写 性 别 者 <asp:Label ID="unknown" runat="server" Text="Label"></asp: 
Label> 人 。 

</div> 

</form> 


(4) 为 “统计 用 户 数 ”按钮 编写 代码 ， 执 行 查询 并 显示 结果 。 代 码 如 下 : 


protected void Button1l Click(object sender, EventArgs e) 
. 


// 从 配置 文件 中 得 到 连接 字符 串 

string s = ConfigurationManager .ConnectionStrings ["database"] . 
Connectionstring; 

SqlConnection connection = new SqlConnection(s); // 创 建 连接 对 象 
string sql = "select count (*) from LoginUser"; 


// 构 建 SQL 语句 (查询 总 用 户 数 ) 
SqlCommand command = new SqlCommand (sql，connection); // 创 建 命令 对 象 
try 
{ 
connection.Open(); // 打 开 连 接 
int n = Convert.ToInt32 (command.ExecuteScalar ()); 
// 执 行 命令 ， 得 到 查询 结果 
total.Text = n.Tostring(); 
// 修 改 SQL 语句 ， 查 询 性 别 为 空 的 用 户 
command.CommandText = "select count(*) from LoginUser where Sex is 
rk eg 
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n = Convert.ToInt32 (command-ExecuteScalar() ); // 执 行 查询 ， 并 显示 结果 
unknown .Text = n.ToString(); 


// 修 改 SQL 语句 ， 根 据 性 别 为 M 或 下 进行 查询 


command .CommandText = "select count (*) from LoginUser where Sex= 


Q@sex"; 


// 为 查询 参数 赋值 为 M， 查 询 男 用 户 数 


SqlParameter pl = new SqlParameter("@sex", 'M'); 


command.Parameters.Add (p1); 

= Convert.ToInt32 (command.ExecuteScalar ()); 
male.Text = n.Tostring(); 
// 为 查询 参数 赋值 为 F， 查 询 男 用 户 数 
command.Parameters[0] .Value = "FEF"; 

= Convert.ToInt32 (command .ExecuteScalar ()); 
female.Text = n.ToString(); 


! 


catch (Exception ex) 


{ 


string message = "执行 命令 过 程 中 出 错 。" 
+ ex.Message.Replace("\r", "\\r ") .Replace("\n"，"N\N\n") 7 
Response.Write (message) 7 


} 
finally 
{ 


command.Dispose(); 
connection.Close(); 


(5) 运行 UserStatistics.aspx 页 面 ， 运 行 结果 如 图 2.11 所 示 。 


用 户 信息 统计 ~ Wezills Firefex 
A 


Be wt - esp 


Gos [1"%:®- 
| 用 户 信息 统计 区 | 


进行 统计 

各 用 户 数 ， 6 

其 中 男 用 户 3 人 ， 女 用 户 1 人 ， 未 填写 性 别 者 2 人 。 
ED EE 


图 2.11 用 户 统计 页 面 


2.4.2 ”数据 读 取 器 DataReader 


DataReader 以 只 


数据 时 ， 只 能 按 一 


读 、 单 向 的 方式 访问 数据 库 。 只 
数据 库 中 检索 数据 ， 而 不 可 以 对 数据 库 的 数据 进行 修改 ; 单 向 ， 指 的 是 月 
定 的 顺序 逐 行 读 取 数据 ， 不 能 


读 ， 指 的 是 使 用 DataReader 只 可 以 从 


日 DataReader 读 取 


回头 读 取 已 经 被 读 取 过 的 数据 。 


在 NET Framework 中 ,实现 DataReader 的 基 类 是 System.Data.Common.DbDataReader。 
针对 不 同 的 数据 库 系 统 ， 从 DbDataReader 类 派生 的 类 有 ObdeDataReader、OleDbData- 
Reader、OracleDataReader SqlDataReader 等 。 下 面 以 SqlDataReader 为 例 说 明 DataReader 


的 使 月 


SqlDataReader 类 的 主要 属性 见 表 2-4。 
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表 2-4 SqlDataReader 类 主要 属性 


属 性 
FieldCount 
HasRows 

IsClosed 


说 明 
当前 行 中 的 列 数 

DataReader 中 是 否 包含 行 
DataReader 是 否 已 经 关闭 


SqlDataReader 类 声明 了 两 个 索引 器 ， 可 以 用 int 类 型 的 列 序号 或 者 string 类 型 的 列 名 
为 索引 。 两 个 索引 器 的 定义 如 下 : 


public override object this [int columnOrdinal] {get;} 


// 根 据 列 序号 取得 DataReader 中 某 列 的 值 


public override object this [string columnName] {get;} 


// 根 据 列 序号 取得 DataReader 中 某 列 的 值 


全 提示 : 若 要 得 到 一 个 SqlDataReader 对 象 , 必须 调用 SqlCommand 对 象 的 ExecuteReader() 
方法 ， 而 不 能 调用 SqlDataReader 的 构造 函数 。SqlDataReader 类 没有 公共 构造 
函数 。 


SqlDataReader 类 的 主要 方法 如 下 : 
(1) 操作 SqlDataReader。 


public void Close() // 关 闭 SqlDataReader 

public virtual bool Read(); // 使 SqlDataReader 前 进 到 下 一 条 记录 
public virtual bool NextResult () // 使 SqlDataReader 前 进 到 下 一 个 结果 
(2) 获取 列 信息 。 

public virtual Type GetFieldType (int i); // 得 到 指定 列 的 数据 类 型 
public virtual string GetName (int i); // 得 到 指定 列 的 列 名 

public virtual int GetOrdinal (string name); // 得 到 具有 指定 名 称 的 列 的 序号 
public override bool IsDBNull(int i); // 判 断 指定 列 是 否 为 空 

(3) 获取 列 的 值 。 

public override object GetValue (int i); // 得 到 第 列 的 值 


// 以 下 方法 以 特定 数据 类 型 返回 指定 列 的 值 

public virtual bool GetBoolean (int i); 
public virtual byte GetByte (int i); 
public virtual byte GetByte (int i); 
public virtual char GetChar (int i); 
public virtual DateTime GetDateTime (int i); 
public virtual decimal GetDecimal (int i); 
public virtual double GetDouble (int i); 
public virtual float GetFloat (int i); 
public virtual short GetInt16(int i); 
public virtual int GetInt32 (int i); 
public virtual long GetInt64(int i); 
public virtual string GetString(int i); 


SqlDataReader 每 一 次 读 取 只 从 数据 库 读 取 一 行 数据 ， 因 此 ， 如 果 要 处 理 多 行 数据 ， 则 
在 处 理 多 行 期 间 ， 数 据 库 连 接 必 须 始 终 保持 可 用 。 在 使 用 SqlDataReader 时 ,与 此 
SqlDataReader 相关 联 的 SqlConnection 正信 于 为 SqlDataReader 服务 , 而 无 法 执行 任何 其 他 
操作 。 
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从 SqlDataReader 中 得 到 某 列 数值 有 4 种 方法 ， 代 码 如 下 : 


// 设 reader 为 SqlDataReader 类 的 实例 
string idl = reader.GetString (0); 


// 方 法 一 : 使 用 强 类 型 的 Get 方法 得 到 列 的 值 
string id2 = (string)reader.GetValue (0); 

// 方 法 二 : 使 用 弱 类 型 的 GetValue 得 到 列 的 值 
string id3 = (string)reader[0]; // 方 法 三 : 使 用 索引 器 《〈 列 序号 作 索 引 ) 
string id4 = (string)reader["CustomerID"]; 


// 方 法 四 : 使 用 索引 器 〈 列 名 作 索引 ) 
【 例 2-S】 读 取 数据 。 
本 例 演示 如 何 通过 DataReader 从 数据 库 读 取 数 据 。 
(1) 打开 例 2-4 所 创建 的 项 目 。 
(2) 在 项 目 中 添加 一 个 页 面 DataReaderPagel.aspx。 
(3) 在 页 面 上 放置 一 个 ASPNET 服务 器 端 Table， 用 于 显示 数据 ， 放 置 一 个 Button， 
用 于 查询 数据 。 页 面 代码 如 下 : 


全 注意 : 在 ASP.NET 页 面 上 显示 数据 一 般 并 不 使 用 Table， 而 是 使 用 专门 的 数据 绑 定 控 
件 ， 如 GridView、DataList 等 。 由 于 本 书 在 后 续 章 节 中 才 会 讲解 数据 绑 定 控件 ， 
为 了 不 使 用 后 面 没有 讲 到 的 内 容 ， 本 章 显 示 数 据 时 使 用 Table。 


<form id="forml" runat="server"> 


<div> 
<asp:Button ID="Buttonl" runat="server" Text=" 查 询 " onclick="Buttonl_ 
Click™ /> 
<h3> 查 询 结 果 </h3> 
<asp:Table ID="result" runat="server" BorderStyle="Solid" Border 
Width="1lpx" 


GridLines="Both" > 
</asp:Table> 
</div> 
</form> 


(4) 在 按钮 的 Click 事件 中 读 取 数据 ， 并 显示 在 页 面 上 。 代 码 如 下 : 


protected void Buttonl Click(object sender, EventArgs e) 


// 从 配置 文件 中 得 到 连接 字符 串 

string s = ConfigurationManager .ConnectionStrings ["database"] . 

Connectionstring; 

/ /创建 和 使 用 连接 对 象 

using (SqlConnection connection = new SqlConnection(s)) 

{ 
string sql = "select UserID,UserName, Password, Sex from LoginUser"; 
SqlCommand command = new SqlCommand(sql, connection); 


connection.Open(); // 打 开 连 接 
// 得 到 DataReader 对 象 
using (SqlDataReader reader = command.ExecuteReader ()) 
while (reader.Read()) // 循 环 读 取 数据 
{ 
TableRow row = new TableRow(); // 在 页 面 表 上 添加 一 行 
result.Rows.Add (row); 
// 将 4 个 字段 添加 到 行 中 
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or tint YT = O08 < 3 LF 
lL 
TableCell cell = new TableCell(); 
cell.Text = Convert.ToString (reader[i]); 
row.Cells.Add (cell); 
} 
} 
} 
command.Dispose(); 
connection.Close(); 
} 
» 


(5) 运行 页 面 ， 运 行 结果 如 图 2.12 所 示 。 


BET ox 
文件 四” 折 久 加、 坦 看 历史 名 书生 | 


[< EOE ms 
Co 


1D Mtp://localh—erPagel. asye| +|- 
-| 
查询 结果 
hser0l| 王 大 家 ”Je66666 ] 
user02| 小 红 (999999 
luser03| 小 张 (6598741 
juser04 泥 美 ladfasdfs 
luser05| 计 算 机 爱好 者 laaaaa 
luser06| 用 户 和 名 |666666 
于 Fg 


图 2.12 数据 读 取 器 示例 


2.5 数据 集 和 数据 适配器 


.NET Framework 中 的 数据 集 DataSet 是 一 个 内 存 中 的 数据 库 模型 ， 数 据 库 中 的 各 种 元 
素 如 表 、 行 、 列 、 外 键 关系 等 都 可 以 再 映射 到 DataSet， 因 此 使 用 DataSet 可 以 准确 反映 数 
据 库 。 数 据 适 配器 DataAdapter 的 作用 相当 于 数据 库 和 DataSet 之 间 的 一 个 桥梁 , 一 方面 可 
以 把 数据 从 数据 库 读 取出 来 放 在 DataSet 中 , 另 一 方面 还 可 以 把 DataSet 中 被 修改 的 数据 写 
回 数据 库 。 本 节 将 介绍 DataSet 和 DataAdapter 的 使 用 。 


2.5.1 数据 集 DataSet 概述 


.NET Framework 中 的 DataSet 类 就 是 一 个 数据 库 的 模型 ， 数 据 库 中 包含 的 许多 元 素 在 
DataSet 中 都 有 对 应 的 类 。DataSet 中 可 以 有 数据 表 和 视图 ， 表 之 间 有 关系 ， 表 中 有 行 和 列 ， 
表 中 每 一 列 都 有 一 种 特定 的 数据 类 型 ， 可 以 或 者 不 可 以 为 室 ， 可 以 被 设置 为 主键 等 等 ， 所 
有 这 些 属性 ， 都 使 DataSet 看 起 来 像 是 一 个 数据 库 。DataSet 对 象 模型 如 图 2.13 所 示 。 

数据 表 DataTable、 数 据 行 DataRow 和 数据 列 DataColumn 是 DataSet 体系 中 最 基本 的 
组 成 元 素 。DataTable 表示 DataSet 中 一 个 数据 表 ，DataColumn 表示 DataTable 中 的 一 列 ， 
DataRow 表示 DataTable 中 的 一 行 。DataTable、DataColumn、DataRow 类 之 间 的 关系 与 数 
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据 库 中 表 、 列 、 行 之 间 的 关系 相同 。 


DataRelationCollection| 


hildRelations 


ParentRelations 


[ExtendedProperties 


lIDataColumnCollection 


ExtendedProperties 


图 2.13 DataSet 对 象 模型 


2.5.2 ”数据 适配器 DataAdapter 概述 


DataAdapter 的 作用 相当 于 数据 库 和 DataSet 之 间 的 一 个 桥梁 ， 一 方面 可 以 把 数据 从 数 


据 库 读 取出 来 放 在 DataSet 中 ， 另 一 方面 还 可 以 把 DataSet 中 被 修改 的 数据 写 回 数据 库 。 这 


能 是 通过 SqlDataAdapter 类 的 4 个 SqlCommand 属性 实现 的 ， 代 码 如 下 : 


//SelectCommand 用 于 从 数据 库 读 取 数据 到 DataSet 

public SqlCommand SelectCommand{get; set;} 
//InsertCommand 用 于 把 DataSet 中 的 新 增 行 插入 到 数据 库 中 
public SqlCommand InsertCommand{get; set;} 
//DeleteCommand 用 于 把 DataSet 中 被 删除 的 行 从 数据 库 中 删除 
public SqlCommand DeleteCommand{get; set;} 
//UpdateCommand 用 于 把 DataSet 中 数据 的 修改 更 新 到 数据 库 中 
public SqlCommand UpdateCommand{get; set;} 


SqlDataAdapter 的 构造 函数 如 下 : 


// 默 认 构 造 函数 

public SqlDataRAdapter (); 

// 创 建 SqlDataAdapter 新 实例 ， 并 用 指定 的 SqlCommand 作为 其 SelectCommand 

public SqlDataAdapter (SqlCommand selectCommand); 

// 创 建 SqlDataAdapter 新 实例 ， 并 指定 其 SelectCommand 的 连接 和 命令 文本 

public SqlDataAdapter(string selectCommandText, SqlConnection select 
Connection); 

// 创 建 SqlDataAdapter 新 实例 ， 并 指定 其 SelectCommand 的 连接 字符 串 和 命令 文本 
public SqlDataAdapter (string selectCommandText, string selectConnection 
String); 


2.5.3 填充 数据 


使 用 SqlDataAdapter 类 的 Fil0 方 法 可 以 从 数据 库 中 读 取 数据 并 填充 到 DataSet 中 。 使 
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用 SqlDataAdapter.Fill0 方 法 填充 数据 时 ， 将 执行 SqlDataAdapter 的 SelectCommand， 并 将 
检索 到 的 数据 填充 到 DataSet 的 一 个 DataTable 中 。 

SqlDataAdapter 类 的 Fill0 方 法 有 多 个 重 载 版 本 ， 常 用 的 有 如 下 3 个 版 本 。 

public override int Fill (DataSet dataset); 


// 填 充 DataSet 中 名 为 Table 的 DataTable 
pubic override int Fill (DataTable table); // 填 充 指 定 的 DataTable 
pubic override int Fill (DataSet dataset, string tableName); 


// 填 充 具有 指定 名 称 的 DataTable 


在 数据 填充 时 ， 如 果 DataSet 中 不 存在 指定 名 称 的 表 ， 则 创建 一 个 具有 指定 名 称 的 表 ， 
并 把 数据 填充 其 中 。 如 果 DataSet 中 已 经 存在 相同 名 称 的 DataTable， 则 可 分 为 两 种 情况 ， 
一 种 情况 是 DataTable 有 主键 ， 那 么 将 根据 主键 匹配 刷新 相应 的 数据 ; 另 一 种 情况 是 
DataTable 没有 主键 ， 则 在 DataTable 中 原 有 行 的 后 面 追加 新 检索 到 的 行 。 


全 注意 : DataSet 中 DataTable 的 名 称 与 数据 库 中 的 表 名 没有 必然 联系 ， 二 者 可 以 相同 也 可 
以 不 同 。 当 使 用 SqlDataAdapter 填充 DataTable 时 ， 如 果 没 有 指定 即将 填充 的 表 
名 ， 则 表 名 默认 为 Table， 并 不 会 自动 使 用 SELECT 语句 中 的 表 名 。 


在 调用 SqlDataAdapterFil() 方 法 以 前 ， 并 不 要 求 所 使 用 的 连接 已 经 打开 ， 但 要 求 连接 
必须 是 合法 的 。 如 果 在 Fil0 方 法 以 前 连接 是 关闭 的 ， 那 么 Fil0 方 法 将 打开 连接 ， 读 取 和 
填充 数据 ， 然 后 关闭 连接 。 如 果 在 Fil0 方 法 以 前 连接 是 打开 的 ， 那 么 Fil0 方 法 后 连接 仍 
然 打 开 。 

用 DataAdapter.Fill0 方 法 填充 DataSet 的 整个 过 程 ， 如 图 2.14 所 示 。 


从 数据 库 
谈 取 数据 


默认 表 名 为 
Table 


添加 新 表 并 i 东 
人 
追加 数据 
到 表 中 
| 
根据 主键 匹 
C 更 新 表 


图 2.14 填充 DataSet 过 程 
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【 例 2-6】 用 DataAdatper 填充 数据 。 

(1) 打开 例 2-4 所 创建 的 项 目 。 

(2) 在 项 目 中 添加 一 个 页 面 FilData.aspx， 在 页 面 上 放置 一 个 Button 和 一 个 Table 以 
查询 和 显示 数据 。 页 面 代 码 如 下 : 


<form id="forml" runat="server"> 


<div> 
<asp:Button ID="Buttonl"” runat="server” Text=" 填 充 数据 " onclick= 
"Battonl Click” /> 
<asp:Table ID="Tablel" runat="server" BorderStyle="Solid" BorderWidth= 
"lpxn 


GridLines="Both"> 


</asp:Table> 
</div> 


</form> 


(3) 为 “填充 数据 ”按钮 的 Click 事件 编写 如 下 代码 。 


protected void Buttonl Click(object sender, EventArgs e) 


{ 


string s = System.Configuration.ConfigurationManager.ConnectionStrings 


["database"] .ConnectionString7 


} 


SqlConnection connection = new SqlConnection(s); 
string sql = "select * from loginuser"; 
SqlDataAdapter adapter = new SqlDataAdapter (sgl, s); 
// 创 建 DataAdapter 

DataSet ds = new DataSet (); // 创 建 一 个 DataSet 对 象 
adapter.Fill (ds); // 填 充 DataSet 
DataTable table = ds.Tables[0]; 
TableHeaderRow header = new TableHeaderRow(); 
// 输 出 表 头 〈 列 名 ) 
foreach (DataColumn column in table.Columns) 
1| 

TableHeaderCell cell = new TableHeaderCell (); 

cell.Text = column.ColumnName; 

header.Cells.Add (cell); 


Tablel .Rows.Add (header); 
// 逐 行 输 出 数据 
foreach (DataRow row in table.Rows) 
{ 
TableRow r = new TableRow(); 
// 逐 列 输出 数据 
for (int i = 0; i < table.Columns.Count; i++) 
{ 
TableCell cell = new TableCell]l (); 
cell.Text = Convert.ToString (row[i]); 
r.Cells.Add (cell); 
} 
Tablel .Rows.Add (r); 
' 


table.Dispose(); 
ds.Dispose(); 
adapter .Dispose(); 


(4) 运行 FillData.aspx 页 面 ， 运 行 界面 如 图 2.15 所 示 。 
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文件 四 编辑 人 查看 胃 历史 @) 书签 @) 工具 四 帮助 四 
Sa 本 ge 


[http:fflecalhes---AFillData_aspz| 二 性 


填充 数据 | 
UserID| UserName lPassword| RegisterDate |Sex 
luser01 | 王 大 家 666666 |2008-02-06 00:00:00 
luser02 | 小 红 999999 “|2008-06-05 00:00:00F 
user03 | 小 张 6598741_|2009-05-02 00:00:00IL 
user04 上 老 美 adfasdfs |2009-08-02 00:00:00| 
luser05 | 计算 机 爱好 者 laaaaa |2009-10-27 00:00:00l 
luser06 | 用 户 名 666666 |2009-05-06 00:00:00| 
E23 EdEk3D 


图 2.15 DataAdapter 填充 数据 


2.5.4 批量 更 新 数据 


DataAdapter 与 数据 库 之 间 的 连接 是 双向 的 , 既 可 以 从 数据 库 读 取 数 据 , 也 可 以 向 数据 
库 更 新 数据 。SqlDataAdaper.Update 方法 的 功能 就 是 把 DataSet 中 发 生 的 变化 写 回 到 数据 库 
中 。 当 SqlDataAdaper.Update 方法 执行 时 , 会 检查 数据 集中 每 一 行 的 状态 (DataRowState)， 
根据 行 的 状态 是 Added、Deleted 还 是 Modified, 分 别 调用 SqlDataAdaper 的 InsertCommand(、 
DeleteCommand() 和 UpdateCommand() 方 法 ， 实 现 数 据 的 更 新 。 如 果 某 行 的 状态 为 
Unchanged， 则 不 会 把 此 行 更 新 到 数据 库 。 

SqlDataAdapter.Update 有 多 个 重 载 版 本 ， 入 口 参 数 以 某 种 方式 指定 了 用 于 更 新 的 数据 
行 集合 ， 返 回 值 是 成 功 更 新 的 行 数 。SqlDataAdapter.Update 常用 有 以 下 版 本 。 


// 根 据 DataRow 数组 中 每 行 的 状态 ， 调 用 相应 的 命令 更 新 数据 库 
public int Update (DataRow[] rows); 


// 更 新 DataSet 中 所 有 发 生变 化 的 行 
public override int Update (DataSet dataset) 


// 更 新 DataTable 中 所 有 发 生变 化 的 行 
public int Update (DataTable table) 


// 更 新 DataSet 中 具有 指定 名 称 的 DataTable 中 发 生变 化 的 行 

public int Update (DataSet dataset, string tableName); 

SqlDataAdapterUpdate() 方 法 通过 检查 DataRow 的 状态 判断 需要 对 数据 库 执行 什么 操 
作 ， 而 不 是 用 DataRow 中 的 数据 更 新 数据 库 中 相应 的 记录 。 举 例 来 说 ,假设 数据 从 数据 库 
填充 到 DataSet 以 后 ， 未 发 生 任何 变化 ， 如 果 人 为 将 某 一 DataRow 的 状态 设置 为 Added， 
那么 调用 SqlDataAdapterUpdate() 方 法 时 ， 就 会 尝试 把 这 一 行 插 入 到 数据 库 中 , 虽然 这 一 行 
数据 并 不 是 新 增 的 ， 数 据 库 中 已 经 存在 完全 相同 的 记录 。 

在 对 数据 库 进行 更 新 时 ，SqlDataAdapter 类 的 UpdateCommand、InsertCommand 和 
DeleteCommand 通常 需要 使 用 DataRow 中 的 数据 作为 参数 。 

以 插入 数据 为 例 ，SqlDataAdapter 类 的 InsertCommand 要 把 DataRow 中 各 列 值 插入 到 
数据 库 表 中 ， 所 以 InsertCommand 必须 带 参数 ， 具 有 类 似 以 下 形式 。 


insert into TableName (Column1l,Column2,) values (@valuel,@value2,...); 
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其 中 @valuel、@value2 参数 的 值 应 该 来 自 要 插入 到 数据 库 中 的 DataRow 的 相应 列 。 
如 果 不 使 用 SqlDataAdapter， 那 么 应 该 用 类 似 如 下 的 代码 实现 插入 过 程 。 


// 这 段 代 码 演示 如 何 把 DataTable 的 数据 插入 到 数据 库 中 

DataTable table; // 要 更 新 的 数据 表 

// 此 处 省 略 设置 table 及 其 数据 的 代码 

//insert 为 要 插入 时 执行 的 SqlCommand 

SqlCommand insert = new SqlCommand( 
"insert into TableName (Column1l,column2) values (@parl,@par2)", 
connection); 

//insert 语句 需要 两 个 参数 

SqlParameter parl = new SqlParameter(); 

parl.ParameterName = "@parl"; 

insert.Parameters.Add (parl); 

SqlParameter par2 = new SqlParameter(); 

par2.ParameterName = "@par2"; 

insert.Parameters.Add (par2); 

// 针 对 DataTable 中 的 DataRow 循环 

foreach (DataRow row in table.Rows) 


// 如 果 行 状态 的 adaded， 则 设置 相应 的 参数 ， 执 行 插入 命令 
if (row.RowState == DataRowState.Added) 
{ 

parl.Value = row["column1"]; 

par2.Value = row["column2"]; 

insert .ExecuteNonQuery (); 


时 


SqlDataAdapter 的 Update() 方 法 提供 了 批量 更 新 数据 的 功能 ， 可 以 用 更 简洁 的 方法 实 
现 与 上 述 代码 相同 的 功能 。SqlParameter 有 一 种 构造 函数 ， 可 以 指定 参数 所 对 应 的 列 ， 从 
而 可 以 在 SqlDataAdapter 的 Update0 方 法 中 自动 用 相应 列 的 值 作为 参数 的 值 。SqlParameter 
类 的 可 以 指定 参数 来 源 的 构造 函数 如 下 : 

public SqlParameter (string paramName, SqlDbType dbType, int size, string 

sourceColumn); 

上 述 构 造 函 数 中 ，paramName 为 参数 名 称 、dbType 为 参数 类 型 、size 为 参数 大 小 、 
sourceColumn 为 参数 来 源 列 的 名 称 。 

SqlCommandCollection 类 也 提供 了 类 似 的 方法 ， 如 下 代码 所 示 。 

public SqlParameter Add (string paramName, SqlDbType dbType, int size, string 
sourceColumn); 

通过 使 用 这 种 自动 从 列 中 取 值 的 参数 ， 结 合 SqlDataAdapter 类 的 Update0 方 法 ， 就 可 
简洁 的 代码 实现 数据 的 批量 更 新 。 

通过 为 SqlDataAdapter 设置 四 个 命令 SelectCommand、InsertCommand、DeleteCommand 
和 UpdateCommand 可 以 实现 更 新 数据 库 ， 但 这 样 做 比较 繁琐 。 当 表 中 的 列 比较 多 时 ， 工 
作 量 就 更 大 。NET Framework 中 提供 了 一 个 SqlCommandBuilder 类 ， 可 以 在 一 定 程度 上 减 
轻 更 新 数据 库 时 的 工作 量 。 

SqlCommandBuilder 类 可 以 为 SqlDataAdapter 自动 生成 单 表 的 更 新 命令 。 所 谓 单 表 ， 
就 是 在 查询 和 修改 中 只 涉及 一 个 数据 库 表 。 如 果 SqlDataAdapter 的 查询 是 多 表 连 接 查 询 ， 
则 SqlCommandBuilder 不 能 生成 任何 更 新 命令 。 另 外 ，SqlCommandBuilder 还 要 求 在 查询 


以 
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中 必须 返回 主键 列 或 者 唯一 列 ， 因 为 SqlCommandBuilder 将 基于 这 些 列 生 成 更 新 命令 。 

如 果 为 SqlDataAdapter 指定 了 SelectCommand， 并 且 把 一 个 SqlCommandBuilder 对 象 
与 SqlDataAdapter 相关 联 ， 那 么 SqlCommandBuilder 将 自动 生成 SqlDataAdapter 的 未 定义 
的 更 新 命令 。 通 过 给 SqlCommandBuilder 的 构造 函数 中 传递 一 个 SqlDataAdapter 参数 ， 或 
者 设置 SqlCommandBuilder 的 DataAdapter 属性 ， 都 可 以 将 SqlCommandBuilder 与 
SqlDataAdapter 关联 起 来 ， 从 而 可 以 为 SqlDataAdapter 自动 生成 命令 。 

【 例 2-7】 批量 更 新 数据 。 

本 例 演示 通过 DataAdapter 和 DataTable 向 数据 库 批量 更 新 数据 。 

(1) 打开 例 2-6 所 创建 的 项 目 。 

(2) 在 项 目 中 添加 一 个 新 页 面 BatchUpdate.aspx。 

(3) 在 页 面 上 添加 一 个 Table 和 一 个 Button。 页 面 代码 如 下 : 


<form id="form1"” runat="server"> 

<div> 

<asp:Button runat="server" ID="update" Text=" 保 存 修改 " onclick="update 
ac /> 3 

<asp:Table runat="server" ID="grid" ></asp:Table> 


</div> 

</form> 

(4) 在 页 面 后 台 的 C# 代 码 中 ， 添 加 一 个 页 面 级 别 的 变量 ， 存 储 检索 到 的 数据 。 
private DataTable data = null; // 查 询 出 的 数据 


(5) 在 页 面 的 Init 事件 中 加 载 数据 。 代 码 如 下 : 
protected override void OnInit (EventArgs e) 


base.OnInit (e); 

data = loadData (); 
} 
private DataTable loadData() 
{ 

// 从 配置 文件 中 获取 连接 字符 串 

string s = 

System.Configuration.ConfigurationManager.Connectionstrings 

["database"] .Connectionstring; 

SqlConnection connection = new SqlConnection(s); // 创 建 连接 

string sql = "select top 5 UserID, UserName from loginuser"; 

SqlDataAdapter adapter = new SqlDataAdapter (sql, s); 

// 创 建 DataAdapter 

DataTable table=new DataTable(); // 创 建 一 个 DataTable 

adapter.FillSchema (table, SchemaType.Source); // 填 充 表 架构 

adapter.Fil]l (table); 

adapter.Fill (table); // 填 充 DataTable 

adapter.Dispose(); 

return table; 


} 
(6) 在 页 面 的 Load 事件 中 显示 数据 。 代 码 如 下 : 


protected void Page Load (object sender, EventArgs e) 


{ 
showData (data); 
上 
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// 显 示 DataTable 中 的 数据 
private void showData (DataTable table) 

int num = table.Columns.Count; // 得 到 总 列 数 

// 输 出 表 头 

TableHeaderRow header = new TableHeaderRow(); 

For (int Es 07 < Don FE) 

{ 
// 在 HTML table 中 循环 输出 各 个 字段 名 称 
TableHeaderCell cell = new TableHeaderCell(); 
cell.Text = table.Columns[i] .ColumnName; 
header .Cells.Add (cell); 

} 

grid.Rows.Add (header); 

// 输 出 表 中 数据 

foreach (DataRow row in table.Rows) 

{ 
// 针 对 数据 表 中 每 一 行 数 据 在 HTML 表格 中 创建 一 行 
// 根 据 数 据 表 中 每 一 个 字段 在 HTML 表格 中 设置 一 个 单元 格 的 内 容 
TableRow LI = new TableRow(); 
string id = Convert.ToString (row[0]); // 得 到 用 户 ID 
string name = Convert.ToString (row[1]); // 得 到 用 户 名 
TableCell cell = new TableCell]l (); 
cell.Text = id; 
r.Cells.Add (cell); 
cell = new TableCell (); 
TextBox 七 = new TextBox(); 
七 -ID = "Eoxtbox™ + Td 
Text = names 
t.TextChanged += new EventHandler(t TextChanged); 
cell.Controls.Add (t); 
r.Cells.Add (cell); 
grid.Rows.Add (r); 


} 


在 上 述 代 码 中 ， 用 一 个 动态 创建 的 TextBox 显示 用 户 名 称 。TextBox 中 的 文字 被 修改 
后 ， 会 触发 TextChanged 事件 ， 在 此 事件 中 修改 DataTable 中 的 数据 ， 以 便 存储 到 数据 库 。 


TextBox 控件 的 TextChanged 事件 处 理 程序 如 下 : 


void 七 TextChanged (object sender, EventArgs e) 
{ 


TextBox t = sender as TextBox; 
TableRow r = 七 .Parent.Parent as TableRow; // 得 到 文本 框 所 在 行 


string id=r.Cells[0] .Text; // 得 到 文本 框 所 对 应 的 用 户 ID 


// 找 到 DataTable 中 对 应 此 ID 的 DataRow 
DataRow row = data.Rows.Find(id); 
if (row == null) return; 


row[1] = t.Text; // 修 改 DataRow 中 的 用 户 名 


} 
(7) 在 “保存 修改 ”按钮 的 Click 事件 中 ， 编 写 代码 保存 。 代 码 如 下 。 


protected void update Click(object sender, EventArgs e) 
{ 


// 从 配置 文件 中 读 取 连接 字符 串 
string s = 
System.Configuration.ConfigurationManager.ConnectionSstrings 
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["database"] .ConnectionString7 


SqlConnection connection = new SqlConnection(s); // 创 建 连接 
String sql = "select top 5 UserID, UserName from loginuser"; 
// 编 写 SQL 语句 


SqlDataAdapter adapter = new SqlDataAdapter (sql, s); / /创建 数 据 适 配器 


// 创 建 CommandBuilder 对 象 以 自动 生成 数据 更 新 命令 
SqlCommandBuilder builder = new SqlCommandBuilder (adqapter) 


adapter.Update (data) // 更 新 数据 
builder.Dispose(); 
adapter.Dispose(); 

} 


(8) 运行 BatchUpdate.aspx 页 面 ， 修 改 多 个 用 户 名 并 保存 ， 数 据 能 够 保存 到 数据 库 中 。 
运行 界面 如 图 2.16 所 示 。 


批量 更 新 数据 - Norills ize =|D| xx| 
文件 四 ”编辑 EF) 查看) 历史 G) 书签 @) 工具 


KE x ea - [al| 
EL | | 
批量 更 新 数据 | 

保存 修改 


UserID UserName 


user01 下 家 
user02 [rm 
user03 [rs 
user04 卫 的 称 
user05 拷 算 机 爱好 者 


图 2.16 批量 更 新 数据 


2.6 存储 过 程 


各 种 大 型 关系 数据 库 都 支持 存储 过 程 。 存 储 过 程 是 一 组 SQL 语句 的 集合 ， 相 当 于 C# 
语言 中 的 方法 。 存 储 过 程 在 效率 、 安 全 性 、 可 复 用 方面 比 普通 的 SQL 语句 有 优势 ， 因 此 在 
实际 项 目 中 也 有 广泛 应 用 。 


2.6.1 调用 存储 过 程 


从 ADONET 的 角度 来 看 ， 存 储 过 程 也 是 一 种 数据 库 命令 (DbCommand)， 只 是 这 个 
DbCommand 的 CommandType 属性 为 StoredProcedure 而 非 默认 的 Text。 在 ADO.NET 中 调 
用 存储 过 程 与 执行 普通 的 SQL 命令 方法 类 似 。 

【 例 2-8】 调用 存储 过 程 修改 数据 。 

本 例 将 创建 一 个 存储 过 程 以 添加 一 个 用 户 ， 并 从 ADO.NET 中 调用 此 存储 过 程 。 

(1) 创建 一 个 ASPNET 应 用 程序 ， 在 项 目 App_Data 文件 夹 中 添加 一 个 数据 库 ， 在 数 
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据 库 中 添加 一 个 表 LoginUser， 表 结构 如 下 。 
UserID: 登录 用 户 ID，nvarchar(10) 类 型 ， 不 可 为 空 ， 主 键 。 
UserName: 用 户 名 称 ，nvarchar(20) 类 型 ， 不 可 为 空 。 
Password: 登录 密码 ，nvarchar(20) 类 型 ， 不 可 为 空 。 
RegisterDate: 用 户 注 册 日 期 ，datetime 类 型 ， 可 空 。 


Sex 


: 性 别 ，nchar(1) 类 型 ， 可 空 ，M 为 男 , 下 为 女 。 


(2) 在 数据 库 中 创建 一 个 存储 过 程 AddUser， 这 个 存储 过 程 接受 用 户 信 息 作 为 参数 ， 


把 用 户 信息 添加 到 数据 库 中 。 存 储 过 程 代码 如 下 : 


CREATE PROCEDURE dbo.AddUser 


AS 


/* 以 下 5 个 变量 为 存储 过 程 参 数 */ 
Qid nvarchar (10), 

@name nvarchar (20) ， 

Qpass nvarchar (20), 

@sex nchar (1) 


insert into LoginUser (UserId, UserName, Password, RegisterDate, Sex) 
values (@id, @name, @pass, getdate(), Q@sex) 
RETURN 


(3) 在 项 目 中 添加 一 个 页 面 , 在 页 面 上 放置 相应 控件 以 输入 用 户 信息 ， 页面 代 码 如 下 : 


<form id="forml" runat="server"> 
<div> 


用 户 登录 ID: <asp:TextBox ID="TextBoxl" runat="server"></asp:TextBox> 
<br /> 

用 户 名 称 : <asp:TextBox ID="TextBox2" runat="server" ></asp:TextBox><br /> 
登录 密码 : 

<asp:TextBox ID="TextBox3" runat="server" TextMode = "Password" ></asp: 
TextBox><br /> 

性 别 : 

<asp:RadioButton ID="male" runat="server" GroupName="sex" Text=" 男 " 
Checked="true" /> 

<asp:RadioButton ID="female" runat="server" GroupName="sex" Text= 
ne 

<asp:Button ID="Button1l"” runat="server"” Text=" 添 加 用 户 " onclick= 
"Buttonl Click" /><br /> 


</div> 
</form> 


(4) 在 “添加 用 户 ” 按 钮 的 Click 事件 中 编写 代码 。 调 用 存储 过 程 AddUser 把 用 户 数 
据 添 加 到 数据 库 ， 代 码 如 下 : 


protected void Buttonl Click(object sender, EventArgs e) 


{ 


// 从 配置 文件 读 取 连接 字符 串 

string s = ConfigurationManager.ConnectionSstrings["database"]. 
Connectionstring; 

SqlConnection connection = new SqlConnection(s); // 创 建 连接 
SqlCommand command = new SqlCommand("AddUser",connection); 
command.CommandType=CommandType.StoredProcedure; // 设 置 命令 为 存储 过 程 
// 为 存储 过 程 添加 4 个 参数 

SqlParameter pl = new SqlParameter("@id",TextBox].Text); 
command.Parameters.Add (pl1); 

SqlParameter p2 = new SqlParameter("@name", TextBox2.Text); 
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command.Parameters.Add (p2) 

SqlParameter p3 = new SqlParameter("@pass", TextBox3.Text); 
command.Parameters.Add (p3); 

SqlParameter p4=new SqlParameter("@sex",male.Checked?"M":"F"); 
command.Parameters.Add (p4); 

string message=""; 


try 
{ 
connection.Open(); // 打 开 连 接 
int n = command.ExecuteNonQuery (); // 执 行 命令 
if (n == 1) 
message = "注册 成 功 。 用 户 信 息 已 经 保存 在 数据 库 。"; 
else 


message = "数据 未 能 保存 。"; 
} 


catch (Exception ex) 


message = "保存 用 户 信息 过 程 中 出 错 。" 
+ ex.Message.Replace("\r", "\\r ").Replace("\n","\\n"); 
} 
finally 
i 
command.Dispose(); 
connection.Close(); 
} 
Page.ClientScript.RegisterStartupScript (this.GetType(), 
"register", "<script>alert(\""+messaget+"\");</script>"); 


1 


【 例 2-9】 调用 存储 过 程 读 取 数 据 。 

本 例 将 调用 存储 过 程 ， 根 据 用 户 输入 的 关键 字 对 用 户 名 进行 搜索 ， 并 把 搜索 结果 显示 
在 页 面 上 。 

(1) 打开 例 2-8 所 创建 的 项 目 。 

(2) 在 数据 库 中 添加 一 个 新 的 存储 过 程 GetUserByName， 这 个 存储 过 程 接 受 一 个 查找 
关键 字 ， 在 数据 库 中 查找 用 户 名 包含 此 关键 字 的 所 有 用 户 。 存 储 过 程 代码 如 下 : 

CREATE PROCEDURE dbo.GetUserByName 

@name nvarchar (20) 
AS 


select * from LoginUser where UserName like '%'+@namet+'%$s" 
RETURN 


(3) 在 项 目 中 添加 一 个 新 页 面 ， 在 新 页 面 上 放置 相应 的 一 个 TextBox 控件 以 接受 用 户 
输入 ， 放 置 一 个 服务 器 端 Table 以 显示 查询 结果 。 页 面 代码 如 下 。 

<form id="forml" runat="server"> 

<div> 

输入 用 户 名 以 进行 搜索 : <asp:TextBox runat="server" ID="userName" /> 


<asp:Button ID="Button1" runat="server"” Text=" 查 找 用 户 " onclick= 
"Button1 Click" /><br /> 
查找 结果 如 下 <br /> 
<asp:Table ID="result" runat="server" BorderStyle="Solid" BorderWidth= 
"lpx" GridLines="Both"/> 

</div> 

</form> 


(4) 在 “查找 用 户 ” 按 钮 的 Click 事件 中 调用 存储 过 程 执 行 查询 ， 并 将 结果 显示 在 页 
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面 上 的 Table 中 。 代 码 如 下 : 


protected void Buttonl Click(object sender, EventArgs e) 
| 
// 从 配置 文件 读 取 连 接 字符 串 
string s = ConfigurationManager.Connectionstrings["database"]. 
Connectionstring; 
// 创 建 并 使 用 数据 库 连 接 对 和 象 


using (SqlConnection connection = new SqlConnection(s)) 


{ 
/ /创建 命 令 对 象 ， 设 置 为 存储 过 程 ， 并 添加 参数 
SqlCommand command = new SqlCommand ("GetUserByName", connection); 
command.CommandType = CommandType.StoredProcedure; 
SqlParameter p = new SqlParameter("@name", userName.Text); 
command.Parameters.Add (p); 
connection.Open(); 
// 执 行 命 令 ， 读 取 数据 


using (SqlDataReader reader = command.ExecuteReader () ) 


while (reader.Read()) 
{ 
// 将 读 取 到 的 数据 循环 添加 到 HTML 表格 中 
TableRow row = new TableRow(); 
result .Rows.Add (row); 
for (int i = O71 < 37 14+) 
{ 
TableCell cell = new TableCell (); 
cell.Text = Convert.ToString (reader [i]); 
row.Cells.Add (cell); 


} 
} 
command.Dispose(); 
connection.Close(); 


} 
(5) 运行 此 页 面 ， 运 行 结果 如 图 2.17 所 示 。 


存储 过 程 示例 Z( 查询 数据 ) -Worills 了 EGR 二 
文件 中。 蝙 辑 下 ) 查看 历史 (GG) 书签 @) 工具 CD) EZ 


~ ey 
Co EE :EE 


存储 过 程 示例 2 《 查询 煞 笑 》 


输入 用 户 名 以 进行 搜索 ;由 查找 用 户 
查找 结果 如 下 

luser02| 小 红 03|999999 

luser03| 小 张 01|6598741 

user07| 小 小 说 123 


ED 


图 2.17 调用 存储 过 程 查询 数据 


2.6.2 ”输出 参数 


存储 过 程 的 参数 默认 情况 下 为 输入 参数 ， 即 参数 值 由 存储 过 程 调 用 者 传递 给 存储 过 
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程 ， 


如 果 在 存储 过 程 中 对 参数 值 进行 了 修改 ， 不 会 影响 到 参数 调用 者 。 存 储 过 程 参数 还 有 


另外 一 种 类 型 ， 称 为 输出 参数 ， 这 种 参数 可 以 双向 传 值 ， 既 可 以 从 存储 过 程 调 用 者 〈 如 
ASPNET 程序 ) 传递 给 存储 过 程 ， 也 可 以 在 存储 过 程 中 修改 参数 值 ， 并 传递 回调 用 者 。 存 
储 过 程 的 输入 输出 参数 的 概念 与 C# 方 法 中 输入 输出 参数 概念 类 似 。 


使 用 存储 过 程 的 输出 参数 需要 注意 两 点 : 一 是 在 定义 存储 过 程 时， 必须 在 参数 后 面 添 


加 output 关键 字 ， 以 说 明 这 是 一 个 输出 参数 ， 二 是 在 C# 中 调用 带 输出 参数 的 存储 过 程 时 ， 
必须 将 相应 的 SqlParameter 参数 的 Direction 属性 设置 为 ParameterDirection.Output。 


【 例 2-10】 存储 过 程 输出 参数 。 

本 例 将 查询 所 有 注册 用 户 总 数 、 男 用 户 数 和 女 用 户 数 , 利用 存储 过 程 输出 参数 来 实现 。 
(1) 打开 例 2-8 所 创建 的 项 目 。 

(2) 在 数据 库 中 添加 一 个 新 的 存储 过 程 GetUserCount， 代 码 如 下 : 


CREATE PROCEDURE dbo.GetUserCount 
@total int output, 
@male int output, 
@female int output 
AS 
select etotal=count (*) from LoginUser ; 
select male=count (*) from LoginUser where Sex='M' 7 
select female=count (*) from LoginUser where Sex="'F'; 
RETURN 


(3) 在 项 目 中 添加 一 个 新 页 面 OutputParameteraspx， 在 页 面 上 放置 1 个 Button 以 调用 


存储 过 程 ， 添 加 3 个 Label 以 显示 查询 结果 。 页 面 代码 如 下 : 


<form id="forml" runat="server"> 

<div> 

<asp:Button ID="Buttonl" runat="server" Text=" 统 计 用 户 数 " onclick= 
"Button1 Click" /><br /> 

用 户 总 数 为 <asp:Label ID="Labell" runat="server" Text="Label"></asp:Label>, 
其 中 男 <asp:Label ID="Label2" runat="server" Text="Label"></asp:Label>, 

女 <asp:Label ID="Label3" runat="server" Text="Label"></asp:Label>, <br /> 
其 余 用 户 注册 时 未 说 明 性 别 。 

</div> 

</form> 


(4) 在 “统计 用 户 数 ”按钮 的 Click 事件 中 ， 编 写 代码 以 执行 存储 过 程 并 显示 结果 ， 


代码 如 下 : 


protected void Button1l_Click(object sender, EventArgs e) 

{ 
string s = ConfigurationManager .ConnectionStrings ["database"] 
.Connectionstring; 
using (SqlConnection connection = new SqlConnection(s)) 


{ 
// 创 建 一 个 命令 对 象 ， 设 置 命令 类 型 为 存储 过 程 
SqlCommand command = new SqlCommand ("GetUserCount", connection); 
command.CommandType = CommandType.StoredProcedure; 
// 创 建 参数 ， 并 说 明 参 数 类 型 
SqlParameter pl = new SqlParameter("@total",SqlDbType.Int); 
pl.Direction = ParameterDirection.Output; ”// 定 义 为 输出 参数 
command.Parameters.Add (p1); // 把 参数 添加 到 存储 过 程 
// 以 同样 的 方式 为 存储 过 程 添加 其 他 输出 参数 
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事务 是 数据 库 中 一 个 很 重要 的 概念 ， 


SqlParameter p2 = new SqlParameter("@male", SqlDbType.Int); 
p2.Direction = ParameterDirection.Output; 
command.Parameters.Add (p2); 


SqlParameter p3 = new SqlParameter ("@female", SqlDbType.Int); 


p3.Direction = ParameterDirection.Output; 
command.Parameters.Add (p3); 
connection.Open(); 
command .ExecuteNonQuery (); // 执 行 命令 
// 命 令 执行 完毕 后 ， 命 令 的 输出 参数 中 已 经 包含 结果 
connection.Close(); 
command.Dispose(); 
// 取 得 输出 参数 的 值 
int nl = (int)command.Parameters["@total"] .Value; 
int n2 = (int)command.Parameters["@male"] .Value; 
int n3 = (int)command.Parameters["@female"] .Value; 
// 把 结果 显示 在 页 面 控件 上 
Labell.Text = nl.ToString(); 
Label2.Text = n2.ToString(); 
Label3.Text = n3.ToString(); 


27 事 务 


是 作为 单个 逻辑 工作 单元 执行 的 一 系列 操作 。 事 


务 具有 4 个 重要 属性 ， 原 子 性 (Atomicity)、 一 致 性 (Consistency)、 隔 离 性 (Isolation， 又 
称 独 立 性 )、 持 久 性 (Durability)， 事 务 的 这 4 个 属性 也 可 以 合并 称 为 ACID。 


2.7.1 事务 的 基本 概念 
事务 有 以 下 4 个 基本 属性 。 
口 原子 性 : 事务 必须 是 原子 工作 单元 ， 对 于 其 数据 修改 ， 或 者 全 都 执行 ， 或 者 全 都 


口 


口 持久 性 : 事务 完成 之 后 ， 对 于 系统 的 影响 是 永久 性 的 。 该 修改 即使 出 现 系统 故障 
也 将 一 直 保持 。 
为 了 理解 事务 的 概念 ， 一 个 常用 的 例子 就 是 银行 转账 。 在 银行 转账 过 程 中 ， 从 原 账户 


不 执行 。 


一 致 性 : 事务 在 完成 时 ， 必 须 使 所 有 的 数据 都 保持 一 致 状态 。 在 相关 数据 库 中 ， 
所 有 规则 都 必须 应 用 于 事务 的 修改 ， 以 保持 所 有 数据 的 完整 性 。 事 务 结束 时 ， 所 


有 的 内 部 数据 结构 都 必须 是 正确 的 。 


隔离 性 :由 并 发 事务 所 作 的 修改 必须 与 任何 其 他 并 发 事务 所 作 的 修改 隔离 。 事 务 
识别 数据 时 数据 所 处 的 状态 ， 或 者 是 另 一 并 发 事务 修改 之 前 的 状态 ， 或 者 是 第 二 


个 事务 修改 之 后 的 状态 ， 事 务 不 会 识别 中 间 状 态 的 数据 。 


扣除 金额 ， 向 目标 账户 添加 金额 ， 这 两 个 数据 库 操作 的 总 和 构成 一 个 完整 的 逻辑 过 程 ， 不 
可 拆 分 。 这 个 过 程 被 称 为 一 个 事务 ， 具 有 ACID 特性 。 
SQL Server 中 主要 有 3 处 事务 : 显 式 事务 、 自 动 提交 事务 和 隐 式 事务 ， 其 中 显 式 事务 
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时 刘 
开征 


使 用 ， 而 且 其 操作 比较 典型 ， 本 节 将 以 显 式 事务 为 例 介 绍 事务 的 使 用 。 
事务 具有 开始 和 结束 。 在 SQL Server 中 ， 以 显 式 事务 为 例 ， 事 务 从 Begin Transaction 
作为 开始 。 事 务 的 结束 又 可 分 为 两 种 情况 , 分 别称 为 提交 (Commit) 和 回 深 (Rollback)。 
事务 是 指 让 事务 中 所 有 操作 生效 ， 而 回 滚 事务 是 指 取消 事务 中 所 有 操作 ， 使 数据 库 恢 
事务 开始 前 的 状态 。 

事务 可 以 嵌 套 ， 即 在 一 个 事务 中 还 可 以 再 开始 一 个 妇 


务 。 


3 


县 提示 : 事务 的 提交 与 事务 谋 套 的 层次 是 相对 应 的 ， 即 Commit 语句 只 提交 赃 套 事务 中 该 


层次 的 事务 , 而 Rollback 语句 会 回 滚 所 有 事务 , 不 论 Rollback 语句 处 于 嵌 套 事务 
哪个 层次 。 

【 例 2-11】 在 SQL Server 中 使 用 事务 。 

本 例 演示 在 SQL Server 中 开始 事务 、 提 交 事务 、 回 深 事 务 ， 以 及 翌 套 事务 。 

(1) 打开 SQL Server 2005 Management Studio， 新 建 一 个 数据 库 Test。 

(2) 在 数据 库 中 新 建 一 个 商品 表 Products， 创 建 表 的 SQL 代码 如 下 。 


CREATE TABLE [dbo] . [Products]( 


[ProductID] [nchar] (10) NOT NULL, /* 商 品 编号 */ 
[ProductName] [nvarchar] (20) NOT NULL, /* 商 品名 称 */ 
[Price] [float] NOT NULL, /* 商 品 价格 */ 
[Stock] [float] NOT NULL, /* 商 品 库 存 */ 


CONSTRAINT [PK Products] PRIMARY KEY CLUSTERED 
([ProductID] ASC) ) 


(3) 向 Products 表 中 添加 几 行 数据 ，SQL 代码 如 下 : 


insert into Products (productid, productname, price, stock) 
values ("pO01”, CPU" 42576)3 

insert into Products (productid, productname, price, stock) 
values ('p002',' 内 存 ',180,20); 

insert into Products (productid, productname, price, stock) 
values ('p003',' 显 示 器 ',1100,5); 


(4) 下 面 演示 事务 和 回 深 ， 在 SQL Server 2005 中 编写 以 下 SQL 语句 。 
set nocount on 

begin transaction /* 开 始 事务 */ 

update Products set price=9999 where ProductID='p001" 
delete Products where ProductID='p002" 

print 'Products 数据 为 (在 事务 中 )' 


select * from Products 
rollback transaction  /* 回 滚 事务 */ 
print "Products 数据 为 《事务 回 滚 后 ) " 


Select * from Products 
运行 以 下 语句 ， 得 到 结果 如 下 : 
Products 数据 为 (在 事务 中 ) 


ProductID ProductName Price Stock 
p001 CPU 9999 6 
p003 显示 器 1100 2 


Products 数据 为 (事务 回 深 后 ) 
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ProductID ProductName Price Stock 
P001 CPU 425 6 
p002 内 存 180 20 
p003 显示 器 1100 5 


从 以 上 运行 结果 可 以 看 出 ， 在 事务 中 ， 把 CPU 的 价格 进行 了 调整 ， 而 且 删 除了 P002 
产品 。 后 来 由 于 事务 回 滚 ， 所 做 修改 全 部 撤销 ， 数 据 又 恢复 到 事务 开始 前 的 状态 。 

(5) 下 面 演示 事务 和 提交 。 在 SQL Server 2005 中 编写 以 下 SQL 语句 。 

set nocount on 

begin transaction /* 开 始 事务 */ 


update Products set price=9999 where ProductID="'p001' 


print 'Products 数据 为 (在 事务 中 )' 
select * from Products where ProductID="'p001" 


commit transaction /* 提 交 事 务 */ 


print 'Products 数据 为 《事务 提交 后 ) " 
Select * from Products where ProductID="'p001" 


运行 以 上 语句 ， 执 行 结果 如 下 : 
Products 数据 为 (在 事务 中 ) 


ProductID ProductName Price Stock 
p001 CPU 3999 6 
Products 数据 为 (事务 提交 后 

ProductID ProductName Price Stock 
p001 CPU 9999 6 


从 以 上 运行 结果 看 ， 由 于 事务 进行 了 提交 ， 所 以 事务 中 所 做 的 修改 (CPU 价格 改 为 
9999) 得 以 保存 。 
(6) 下 面 演示 嵌 套 事务 。 在 SQL Server 2005 中 编写 以 下 代码 : 


set nocount on 

print ' 未 修改 以 前 的 原始 数据 ' 

Select * from Products where ProductID="'p001" 

begin transaction  /* 开 始 事 务 */ 

update Products set price=666 where ProductID='p001' 


print 'Products 数据 为 〈 在 顶层 事务 中 ) " 
select * from Products where ProductID="'p001" 


begin transaction ” /* 开 始 嵌 套 事务 */ 
update Products set Stock=100 where ProductID="'p001"' 


print "Products 数据 为 〈 在 嵌 套 事务 中 ) " 
Select * from Products where ProductID="'p001" 


commit transaction  /* 提 示 霸 套 事务 */ 


print "Products 数据 为 〈 媒 套 事务 提交 后 ) " 
select * from Products where ProductID="'p001" 


rollback transaction  /* 回 滚 项 层 事务 */ 


print "Products 数据 为 〈 顶 层 事务 回 滚 后 ) " 
Select * from Products where ProductID="'p001" 


以 上 代码 中 有 两 个 嵌 套 事务 ， 在 顶层 事务 中 把 商品 P001 价格 修改 为 666, 在 网 套 事务 
中 把 商品 P001 的 数量 修改 为 100, 然后 提交 堪 套 事务 , 回 滚 顶 层 事务 , 代码 运行 结果 如 下 : 
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未 修改 以 前 的 原始 数据 

ProductID ProductName Price Stock 
p001 CPU 888 6 
Products 数据 为 (在 顶层 事务 中 ) 

ProductID ProductName Price Stock 
P001 CPU 666 6 
Products 数据 为 〈 在 嵌 套 事务 中 ) 

ProductID ProductName Price Stock 
P001 CPU 666 100 
Products 数据 为 〈 顽 套 事务 提交 后 ) 

ProductID ProductName Price Stock 
P001 CPU 666 100 
Products 数据 为 (顶层 事务 回 深 后 ) 

ProductID ProductName Price Stock 
p001 CPU 888 6 


从 上 述 运 行 结果 可 以 看 出 ， 虽 然 殿 套 事务 被 提交 ， 但 是 由 于 顶层 事务 被 回 深 ， 所 以 霸 
套 事务 中 所 做 的 修改 也 没有 保存 。 


2.7.2 ADO.NET 中 的 事务 


在 ADONET 中 ， 用 System.Data.Common.DbTransaction 类 表示 数据 库 的 事务 。 
DbTransaction 类 是 一 个 抽象 类 ,不 能 被 实例 化 。DbTransaction 类 作为 其 他 具体 数据 库 事务 
类 的 基 类 ,定义 了 公共 接口 。 对 于 SQL Server 数据 库 来 说 ，SqlTransaction 表示 数据 库 的 事 
务 。 

DbConnection 常用 有 以 下 两 个 方法 。 

口 Commit0: 提交 数据 库 事务 。 此 方法 无 参数 和 返回 值 。 

口 Rollback0: 回 滚 数据 库 事务 。 此 方法 无 参数 和 返回 值 。 

DbConnection 及 其 派生 类 没有 公共 构造 函数 , 所 以 不 能 用 new 创建 一 个 DbTransaction 
或 其 派生 类 的 实例 。 要 想得到 DbTransaction 的 一 个 实例 ， 应 该 调用 DbConnection 类 的 
BeginTransaction() 方 法 。DbConnection 类 的 BeginTransaction() 方 法 签名 如 下 。 

public DbTransaction BeginTransaction(); 

一 个 数据 库 的 事务 中 通常 包含 多 个 操作 。 对 于 ADO.NET 来 说 ， 一 个 数据 库 事务 通常 
包含 多 个 数据 库 命令 DbCommand 对 象 。 为 了 将 DbCommand 与 事务 DbTransaction 关联 起 
来 ， 需 要 设置 DbCommand 类 的 Transaction 属性 。DbCommand 类 的 Transaction 属性 签名 
如 下 : 

public DbTransaction Transaction { get; set; } 

【 例 2-12】 ADONET 事务 。 

本 例 演示 如 何在 ADO.NET 中 使 用 数据 库 事务 , 包含 开始 事务 、 提 交 事务 、 回 深 事 务 。 

(1) 创建 一 个 ASPNET Web 应 用 程序 。 

(2) 在 项 目 中 添加 一 个 LoginUser 表 ， 表 结构 同上 一 节 的 例子 。 
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(3) 在 项 目 中 添加 一 个 页 面 AdoNetTran.aspx， 页 面 上 放置 一 个 Button， 在 Button 的 


Click 事件 中 ， 


添加 如 下 代码 : 


protected void Button1l Click(object sender, EventArgs e) 


| 


// 从 配置 文件 读 取 连 接 字 符 串 

string s = ConfigurationManager .ConnectionStrings ["database"] . 
Connectionstring; 

SqlConnection connection = new SqlConnection(s); / /创建 连接 


// 创 建 一 个 命令 对 象 


SqlCommand commandl1 = 


Con 


new SqlCommand ("insert into LoginUser (UserID, UserName, Password, 
Sex) "+" values ('user99',' 新 用 户 ','123','M') ",， connection); 


nection.Open(); 


// 创 建 另外 一 个 命令 对 象 


SqlCommand command2= 


WY 
sq 


new SqlCommand ("insert into LoginUser (UserID,UserName,Password, 
Sex) "+" values ('user100',' 新 用 户 ', '123',' 男 生 ') "， connection); 


上 面 这 条 语句 会 出 错 ， 因 为 性 别 "男生 "为 两 个 字符 ， 超 出 字段 宽度 


Transaction tran=connection.BeginTransaction(); // 开 始 一 个 事务 


// 将 以 上 两 个 命令 添加 到 同一 个 事务 中 
command]1 .Transaction=tran; 
command2.Transaction=tran; 
string message=""; 

ETY 


} 


command] .ExecuteNonQuery (); / /执行 命 令 1 
command2 .ExecuteNonQuery (); // 执 行 命令 2 
tran.Commit (); // 提 交 事 务 

message=" 保 存 成 功 ! "; 


catch (Exception ex) 


{ 


} 


message = "保存 用 户 信息 过 程 中 出 错 。 事 务 回 滚 。"” ，; 
tran.Rollback (); // 发 生 异常 时 事务 回 深 


加 


finally 


{ 


} 


command1 .Dispose(); 
command2 .Dispose(); 
tran.Dispose(); 

connection.Close(); 


Page.ClientScript-RegisterStartupScript (this.GetType(), 


在 上 面 


两 个 命令 使 


库 定义 (性 


令 执 行 的 语句 跳 转 到 catch 块 中 ， 回 滚 事务 。 由 于 两 个 命令 使 用 同一 个 事务 ， 事 务 一 旦 回 


"register", "<script>alert(\""+messaget+"\");</script>"); 


的 代码 中 ， 一 共 执 行 了 两 条 插入 命令 ， 向 LoginUser 表 中 插入 两 个 新 用 户 ， 这 
用 同一 个 事务 。 第 一 条 命令 是 正确 的 ， 而 第 二 条 命令 所 插入 的 数据 不 符合 数据 
别 字段 太 长 超出 字段 宽度 ) 在 执行 时 会 发 生 异 常 。 程 序 在 运行 时 会 从 第 二 条 命 


滚 ， 两 个 命令 所 做 的 改动 都 被 撤销 ， 数 据 库 恢 复 成 命令 执行 前 的 状态 。 
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2.7.3 TransactionScope 类 的 使 用 


.NET 中 还 有 另外 一 种 处 理事 务 的 方式 : 使 用 System.Transactions.TransactionScope 类 。 
TransactionScope 类 在 System.Transactions 程序 集中 ,默认 情况 下 ，ASPNET 应 用 程序 并 不 
包含 对 这 个 程序 集 的 引用 ， 所 以 如 果 使 用 TransactionScope 类 ， 就 需要 手工 添加 对 
System.Transactions 程序 集 的 引用 。 

2.7.2 节 所 讲 的 DbTransaction 类 只 能 用 于 一 个 数据 库 事 务 处 理 , 而 TransactionScope 类 
可 用 于 分 布 式 多 个 数据 库 事务 。 利 用 TransactionScope 类 能 够 创建 可 提升 事务 。 可 提升 事 
务 是 可 以 根据 需要 自动 提升 为 完全 分 布 式 事务 的 轻型 (本 地 ) 事务 。 

TransactionScope 类 只 有 一 个 常用 方法 Complete0 方 法 ， 指 示 事 务 正常 结束 。 如 果 不 调 
用 Complete0 方 法 ， 那 么 当 TransactionScope 对 象 释放 时 ， 将 会 回 滚 事务 中 的 所 有 操作 。 

TransactionScope 类 表示 的 事务 范围 为 从 创建 TransactionScope 类 的 实例 开始 ， 到 调用 
Complete() 方 法 或 者 这 个 TransactionScope 类 的 实例 被 释放 。 

通常 用 一 条 using 语句 来 使 用 TransactionScope 类 时 ， 把 事务 中 包含 的 语句 放 到 using 
的 大 括号 中 ， 在 大 括号 内 最 后 一 条 语句 是 TransactionScope.Complete() 方 法 。 代 码 如 下 : 

using (TransactionScope scope = new TransactionScope()) 


// 这 里 写 执行 事务 中 的 操作 
scope.Complete(); // 结 束 事务 


} 


在 上 述 代 码 中 , 如 果 大 括号 中 的 语句 都 执行 成 功 (未 发 生 异 常 ), 那么 TransactionScope. 
Complete() 方 法 就 会 被 调用 ， 事 务 提 交 ， 和 否则 ， 如 果 大 括号 中 任意 一 条 语句 发 生 异 常 ， 那 
么 程序 就 会 跳 到 catch 中 执行 (有 catch 语句 时 ) 或 者 中 断 (没有 catch 语句 时 )， 
TransactionScope.Complete() 方 法 不 会 被 调用 ， 事 务 回 滚 。 

【 例 2-13】 TransactionScope 示例 。 

本 例 将 向 两 个 数据 库 中 添加 数据 ， 并 且 通 过 TransactionScope 类 把 这 两 个 不 同 数据 库 
的 操作 合并 为 一 个 事务 来 处 理 ， 统 一 提交 或 者 回 滚 。 

(1) 创建 一 个 ASPNET Web 应 用 程序 。 

(2) 在 项 目 中 添加 对 System.Transactions 程序 集 的 引用 。 

(3) 在 项 目 中 添加 两 个 数据 库 ， 名 称 分 别 为 databasel.mdf 和 database2.mdf。 在 两 个 数 
据 库 中 分 别 创建 一 个 Products 表 ， 建 表 SQL 语句 如 下 : 


CREATE TABLE [Products]( 


[ProductID] [nchar] (10) NOT NULL, /* 商 品 编号 */ 
[ProductName] [nvarchar] (20) NOT NULL，/* 商 品名 称 */ 
[Price] [float] NOT NULL, /* 商 品 价格 */ 
[Stock] [float] NOT NULL, /* 商 品 库存 */ 


CONSTRAINT [PK Products] PRIMARY KEY CLUSTERED 
([ProductID] ASC) ) 
(4) 在 项 目 中 添加 一 个 页 面 TransactionScopePage.aspx， 在 页 面 上 放 一 个 Button。 在 
Button 的 Click 事件 中 编写 代码 ， 向 两 个 数据 库 的 Products 表 添 加 数据 ， 并 使 用 
TransactionScope 类 将 两 个 不 同 数据 库 的 插入 命名 组 合 为 一 个 事务 。 代 码 如 下 : 
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全 


protected void Button]l Click(object sender, EventArgs e) 


{ 


// 两 个 连接 字符 串 ， 连 接 到 两 个 不 同 数据 库 

string sl = @"Data Source=.\SQLEXPRESS;AttachDbFilename= 
lIDataDirectory| \Databasel .mdf; Integrated Security=True;User Instance= 
True"™; 

string s2 = @"Data Source=.\SQLEXPRESS;AttachDbFilename= 
IlDataDirectory| \Database2 .mdf; Integrated Security=True;User Instance= 
True"s 

// 使 用 以 上 两 个 连接 字符 串 创建 两 个 连接 

SqlConnection connectionl = new SqlConnection(s1); 

SqlConnection connection2 new SqlConnection(s2); 

// 创 建 一 个 命令 对 象 执行 一 条 insert 语句 


SqlCommand command = new SqlCommand( ); 


command .CommandText="insert into Products (ProductID, ProductName, Price， 


Stock) "+" values ('"P101"，' 新 产品 ,100,20) "; 


using (TransactionScope tran=new TransactionScope ()) 
{ 

string message=""; 

try 


1 
// 让 命令 使 用 第 一 个 连接 ， 更 新 第 一 个 数据 库 
connection]l .Open(); 
command.Connection = connectionl; 
command.ExecuteNonQuery (); // 执 行 命令 
// 让 命令 使 用 第 二 个 连接 ， 更 新 第 二 个 数据 库 
connection2.O0pen(); 
command.Connection = connection2; 


command.ExecuteNonQuery (); / /执行 命 令 
message=" 保 存 成 功 ! "; 
tran.Complete(); // 提 交 事务 
| 
catch (Exception ex) 
{ 
// 如 果 执行 到 catch 块 中 ， 说 明 try 中 有 异常 
// 而 try 的 最 后 一 名 tran.Complete 肯定 没有 成 功 执行 ， 所 以 事务 没有 提交 
message = "保存 用 户 信息 过 程 中 出 错 。 事 务 回 滚 。" ，; 
} 
finally 
{ 
command.Dispose(); 
tran.Dispose(); 
// 检 查 两 个 连接 状态 ， 如 果 打开 则 关闭 
if(connection]l.State==ConnectionState.Open) 
connectionl.Close(); 
if (connection2.State == ConnectionState.Open) 
connection2.Close(); 
} 


Page.ClientScript.RegisterStartupScript (this.GetType(), 
"register", "<script>alert(\""+messaget+"\");</script>"); 


(5) 运行 页 面 ， 单 击 按钮 ， 可 以 看 到 ， 上 述 代码 成 功 将 数据 插入 到 两 个 数据 库 中 。 
(6) 修改 上 述 代码 ， 将 第 二 条 命令 文本 改 成 一 条 错误 SQL 语句 ， 从 而 导致 程序 运行 时 


b 现 异常 。 关 键 代码 如 下 : 
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try 


(7) 


// 让 命令 使 用 第 一 个 连接 ， 更 新 第 一 个 数据 库 

connection]l .Open (); 

command.Connection = connection]l; 

command.ExecuteNonQuery (); // 执 行 命令 

// 让 命令 使 用 第 二 个 连接 ， 更 新 第 二 个 数据 库 

command.CommandText = "insert into Products (price) values (3)"; 
connection2.O0pen(); 

command.Connection = connection2; 


command.ExecuteNonQuery (); // 执 行 命令 
message=" 保 存 成 功 ! "7 

tran.Complete(); // 提 交 事务 

再 次 运行 页 面 ， 单 击 按钮 ， 可 以 看 到 ， 由 于 代码 出 错 ， 事 务 回 滚 ， 两 条 修改 数据 


的 语句 都 被 撤销 。 


2.8 通用 数据 访问 类 SqlHelper 


通过 前 面 几 节 的 介绍 ， 可 以 看 到 用 ADONET 操作 数据 库存 在 大 量 的 重复 工作 。 一 般 
来 说 ，ADO.NET 操作 数据 库 包含 以 下 几 个 步 又: 


(1) 
《2 
(3) 
(4) 
《9 
(6) 
(7) 
(8) 


获取 连接 字符 帅 ， 创 建 到 数据 库 的 连接 。 
打开 连接 。 

创建 命令 对 象 。 

为 命令 添加 参数 。 

执行 命令 ， 获 得 命令 结果 。 

关闭 命令 。 

关闭 连接 。 

处 理 命令 结果 。 


在 上 述 步骤 中 ， 除 了 命令 文本 不 同 ， 执 行 命令 的 方法 和 返回 内 容 不 同 ， 其 他 步骤 都 是 


相同 的 ， 


分 的 重复 工作 ， 从 而 避免 每 次 访问 数据 库 都 要 写 大 量 重复 代码 。 实 际 上 ，ADO.NET 刚 出 


甚至 代码 都 一 样 。 从 软件 设计 的 角度 来 说 ， 应 该 设计 一 个 通用 数据 类 ， 负 责 大 部 


现时 ， 就 有 开发 人 员 写 出 了 这 样 的 通用 数据 访问 类 ， 包 括 微软 公司 自己 也 推出 了 通用 数据 


访问 类 


SqlHelper 类 。 微 软 公司 的 通用 数据 访问 类 经 过 不 断 完善 发 展 ， 现 在 已 经 成 为 了 企 


业 库 EnterpriseLibrary 的 一 部 分 。 
本 节 将 介绍 SqlHelper 类 的 设计 和 实现 。 这 个 SqlHelper 类 的 代码 主要 来 源 网 络 ， 笔 者 
对 其 进行 了 适当 修改 。 从 尊重 知识 产权 角度 来 说 ， 笔 者 需要 给 出 SqlHelper 类 的 原始 作者 


和 链接 ， 
所 以 这 是 


但 是 由 于 SqlHelper 类 在 网 上 有 很 多 转载 和 不 同 版 本 ， 笔 者 无 法 找到 原始 作者 ， 
无 法 注 明 出 处 。 


从 前 面 分 析 的 ADO.NET 操作 数据 库 的 8 个 步骤 来 看 ， 主 要 涉及 3 种 对 象 : 数据 库 连 
接 、 数 据 库 命令 、 命 令 参数 。 通 用 数据 访问 类 的 主要 功能 就 是 管理 这 3 种 对 象 。 
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2.8.1 


管理 连接 


连接 是 ADO.NET 中 最 基础 的 对 象 , 所 有 数据 库 操作 都 要 基于 一 个 数据 库 连 接 。 


数据 访问 类 中 对 数据 库 连 接 的 管理 主要 就 是 获取 连接 字符 串 和 创建 连接 。 相 关 代 码 : 


public class SqlHelper 


/ /数据 提供 程序 

private static string dbProviderName = 
ConfigurationManager.ConnectionStrings ["database"] .Provider 
Name; 

// 连 接 字符 串 

private static string dbConnectionString = 
ConfigurationManager.ConnectionStrings ["database"] .Connection 
String; 

private DbConnection connection; // 连 接 对 象 

#region 构造 函数 

public SqlHelper() 

{ 
this .connection = CreateConnection (SqlHelper.dbCconnectionString) 7 

} 

public SqlHelper(string connectionString) 

| 


} 

#endregion 

#region 创建 连接 

/// <summary> 

/// 根据 配置 文件 中 的 连接 字符 串 创建 数据 库 连接 对 象 
/// </summary> 

/// <returns> 所 创建 的 连接 对 象 </returns> 


public static DbConnection CreateConnection() 


{ 


this .connection = CreateConnection (ConnectionString) ; 


DbProviderFactory dbfactory = DbProviderFactories. GetFactory 
( SqlHelper.dbProviderName); 
DbConnection dbconn = dbfactory.CreateConnection(); 
dbconn.Connectionstring = SqlHelper.dbConnectionSstring; 
return dbconn; 

上 

/// <summary> 

/// 用 指定 的 连接 字符 串 创建 数据 库 连 接 对 象 

/// </summary> 

/// <param name="connectionString"> 连 接 字 符 串 </param> 

/// <returns> 所 创建 的 连接 对 象 </returns> 

public static DbConnection CreateConnection(string connectionstring) 

| 
DbProviderFactory dbfactory = DbProviderFactories.GetFactory 

(SqlHelper.dbProviderName); 

DbConnection dbconn = dbfactory.CreateConnection(); 
dbconn.Connectionstring = connectionstring; 
return dbconn; 

. 


#endregion 
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2.8.2 创建 命令 


创建 好 数据 库 连 接 以 后 ， 接 下 来 的 工作 就 是 创建 数据 库 命令 。 数 据 库 命令 主要 分 为 两 
种 : 文本 命令 和 存储 过 程 。SqlHelper 类 定义 了 两 个 方法 ， 分 别 用 于 创建 这 两 种 命令 。 
#region 创建 命令 


/// <summary> 

/// 创建 存储 过 程 命令 

/// </summary> 

/// <param name="storedProcedure"> 存 储 过程 名 称 </param> 
/// <returns> 所 创建 的 命令 </returns> 


public DbCommand GetStoredProcCommond (string storedProcedure) 


DbCommand dbCommand = connection.CreateCommand () 
dbCommand .CommandText storedProcedure; 

dbCommand .CommandType CommandType.StoredProcedure; 
return dbCommand; 


} 

/// <summary> 

/// 创建 文本 命令 

/// </summary> 

/// <param name="sql"> 命 令 文 本 </param> 

/// <returns> 所 创建 的 命令 </returns> 

public DbCommand GetSqlstringCommond (string sql) 

| 
DbCommand dbCommand = connection.CreateCommand () 
dbCommand .CommandText sql; 
dbCommand.CommandType CommandType.Text; 
return dbCommand; 

} 


#endregion 


2.8.3 ”添加 命令 参数 


ADO.NET 中 的 命令 通常 都 带 有 一 个 或 者 多 个 参数 。 在 原始 的 ADO.NET 代码 中 , 创建 
和 添加 命令 参数 也 是 一 项 很 繁琐 的 工作 ， 有 必要 在 SqlHelper 类 中 添加 相应 的 方法 实现 这 
个 功能 。 在 ADO.NET 中 ， 数 据 库 命令 参数 分 为 两 种 :输入 参数 和 输出 参数 ， 在 编写 方法 
时 也 要 考虑 到 这 两 点 。 


/// <summary> 
/// 为 命令 添加 输出 参数 
/// </summary> 
/// <param name="cmd"> 命 令 </param> 
/// <param name="parameterName"> 参 数 名 </param> 
/// <param name="dbType"> 参 数 类 型 </param> 
/// <param name="size"> 大 小 </param> 
public void AddoutParameter (DbCommand cmd, string parameterName, DbType 
dbType, int size) 
| 
DbParameter dbParameter = cmd.CreateParameter(); 
dbParameter .DbType = dbType; 
dbParameter .ParameterName = parameterName; 
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dbParameter.Size = size; 
dbParameter.Direction = ParameterDirection.Output; 
cmd.Parameters.Add (dbParameter); 
} 
/// <summary> 
/// 为 命令 添加 输入 参数 
/// </summary> 
/// <param name="cmd"> 命 令 </param> 
WA "parameterName"> 参 数 名 </param> 
/// <param name="dbType"> 参 数 类 型 </param> 
/// <param name="value"> 参 数值 </param> 
public void AddInParameter (DbCommand cmd, string parameterName, DbType 
dbType, object value) 
{ 


DbParameter dbParameter = cmd.CreateParameter (); 
dbParameter .DbType = dbType; 
dbParameter .ParameterName = parameterName; 
dbParameter.Value = value; 
dbParameter .Direction = ParameterDirection.Input; 
cmd.Parameters.Add (dbParameter); 

} 

/// <summary> 

/// 根据 参数 名 得 到 参数 

/// </summary> 

/// <param name="cmd"> 命 令 </param> 

/// <param name="parameterName"> 参 数 名 </param> 

/// <returns> 得 到 的 参数 </returns> 

public DbParameter GetParameter (DbCommand cmd, string parameterName) 


{ 
; 


return cmd.Parameters [parameterName]; 


2.8.4 执行 命令 


正如 本 章 前 面 几 节 所 介绍 的 ， 在 ADO.NET 中 ， 数 据 库 命令 根据 查询 结果 不 同 ， 有 几 
种 不 同 的 执行 方法 ， 分 别 是 没有 查询 结果 的 ExecuteNonQuery0 方 法 、 查 询 单 个 值 的 
ExecuteScalar() 方 法 和 查询 多 行 多 列 的 ExecuteReader() 方 法 。 通 用 数据 访问 类 SqlHelper 也 
是 按照 这 种 分 类 方式 对 数据 库 命 令 的 执行 进行 了 封装 。 主 要 代码 如 下 : 


// 执 行 命令 ， 得 到 DataReader 对 象 
public DbDataReader ExecuteReader (DbCommand cmd) 
a 


cmd.Connection.Open (); 


DbDataReader reader = cmd.ExecuteReader (CommandBehavior.Close 


Connection); 
return reader; 


由 


// 执 行 没 有 查询 结果 的 命令 (如 insert，delete) ， 返 回 受 影响 的 行 数 


public int ExecuteNonQuery (DbCommand cmd) 
下 
cmd.Connection.Open(); 
int ret = cmd.ExecuteNonQuery(); 
cmd.Connection.Close(); 
return ret; 
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// 执 行 命令 ， 并 返回 查询 到 的 单个 值 
public object ExecuteScalar (DbCommand cmd) 
1 
cmd.Connection.Open(); 
object ret = cmd.ExecuteScalar(); 
cmd.Connection.Close(); 
return ret; 
/// <summary> 
/// 执行 命令 ,返回 一 个 DataTable 
/// </summary> 
/// <param name="cmd"> 要 执行 的 命令 </param> 
/// <returns> 得 到 的 DataTable</returns> 
public DataTable ExecuteDataTable (DbCommand cmd) 
:1 
DbDataReader reader = ExecuteReader (cmd); 
DataTable table = new DataTable(); 
table.Load (reader); 
reader.Close(); 
return table; 


2.8.5 释放 资源 


数据 库 连 接 是 一 种 宝贵 资源 ， 使 用 完毕 后 应 该 及 时 释放 。 对 于 SqlHelper 类 来 说 ， 应 
该 提供 一 个 显 式 的 释放 数据 库 连 接 的 接口 以 供用 户 调用 ， 同 时 还 应 该 在 垃圾 回收 时 自动 释 
放 数 据 库 连接 。 可 以 通过 IDisposable 接口 来 实现 上 述 功能 。 
IDisposable 接口 中 只 包含 一 个 方法 Dispose0, 在 此 方法 中 完成 对 当前 对 象 所 占用 资源 
的 释放 。 在 实现 IDisposable 接口 时 ， 要 考虑 到 两 种 不 同 的 情况 ， 即 用 户 代 码 调用 Dispose 
方法 释放 资源 和 垃圾 回收 时 释放 资源 。 如 果 用 户 代 码 调用 Dispose0 方 法 ， 则 此 时 对 象 没有 
被 垃圾 回收 ， 对 象 中 的 所 有 资源 〈 包 括 托管 资源 和 非 托管 资源 ) 都 是 可 用 的 ， 应 该 全 部 释 
放 这 些 资源 。 而 当 对 象 被 垃圾 回收 时 ， 对 象 中 的 托管 资源 已 经 被 释放 ， 则 只 需要 释放 非 托 
管 资 源 。 根 据 以 上 讨论 ， 可 以 对 SqlHelper 类 作 以 下 修改 。 
public class SqlHelper:IDisposable 
#region 释放 资源 
private bool disposed = false; / /资源 是 否 已 经 被 释放 
#region IDisposable 成 员 
public void Dispose () 


{ 


dispose (true); 


/// <summary> 


/// 释放 资源 

/// </summary> 

/// <param name="disposing"> 是 否 用 户 代 码 调 用 </param> 

/// <remarks> 

/// 此 方法 将 在 两 种 情况 下 被 调用 ，〈1) 用户 代码 调用 Dispose 方法 () (2) 垃圾 回收 
/// 在 第 2 种 情况 下 ， 数 据 库 连 接 已 经 被 垃圾 回收 自动 处 理 ， 不 需要 再 释放 

/// 只 有 在 第 1 种 情况 下 需要 在 代码 中 显 式 关闭 数据 库 连 接 


/// </remarks> 
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2.8; 


的 功 
加 能 


加 新 


用 户 
面 代 


Private void dispose (bool disposing) 
| 

if (disposed) return; 

disposed = true; 

if (disposing) 

if (connection.State != ConnectionState.Closed) 
connection.Close(); 

} 
#endregion 
#endregion 


6 SqlHelper 应 用 举例 


前 面 介绍 了 SqlHelper 类 的 设计 和 实现 ， 下 面 通过 一 个 实例 来 说 明 其 应 用 。 这 个 例子 
能 与 前 面 所 讲 的 两 个 例子 功能 相同 ， 但 实现 方法 不 一 样 。 对 照 这 两 种 不 同 的 实现 ， 更 
够 体现 SqlHelper 类 所 带 来 的 方便 。 

【 例 2-14】 SqlHelper 示例 。 

本 例 演示 如 何 使 用 通用 数据 访问 类 SqlHelper。 本 例 包括 两 个 功能 : 查询 用 户 列表 和 添 
户 。 其 中 添加 用 户 功 能 通过 存储 过 程 实现 ， 而 查询 用 户 通过 SQL 文本 命令 实现 。 
(1) 创建 一 个 ASPNET 应 用 程序 。 

(2) 在 项 目的 App_Data 文件 夹 中 添加 一 个 数据 库 ， 向 数据 库 中 添加 一 个 LoginUser 
表 结 构 同 前 。 

(3) 在 数据 库 中 添加 一 个 存储 过 程 AddUser。 

CREATE PROCEDURE dbo.AddUser 


@id nvarchar (10), @name nvarchar (20) ,apass nvarchar (20),@sex 
nchar (1) 


RS 
insert into LoginUser (UserId, UserName, Password, RegisterDate, Sex) 
values (@id, @name, @pass, getdate(), @sex) 


(4) 在 项 目 中 添加 一 个 页 面 SqlHelperDemo.aspx。 页 面 上 部 放置 一 个 Table， 用 于 显示 
列表 。 页 面 下 部 放置 几 个 TextBox 和 RadioButton， 用 于 输入 用 户 信息 以 添加 用 户 。 页 
码 如 下 : 


<form id="forml" runat="server"> 

<div> 

<div> 

<h3> 用 户 列表 </h3> 
<asp:Table ID="result" runat="server" GridLines="Both" CellPadding= 
nm2n> 
</asp:Table> 

</div> 

<he /> 

<div> 
<h3> 添 加 新 用 户 </h3> 
用 户 登 录 ID: <asp:TextBox ID="TextBoxl" runat="server"></asp:TextBox> 
<br /> 
用 户 名 称 : <asp:TextBox ID="TextBox2" runat="server" ></asp:TextBox><br /> 
登录 密码 : 


<asp:TextBox ID="TextBox3" runat="server" TextMode="Password"></asp: 
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TextBox><br /> 

性 别 : 

<asp:RadioButton ID="male" runat="server" GroupName="sex" Text=" 男 " 
Checked="true" /> 

<asp:RadioButton ID="female" runat="server" GroupName="sex" Text= 
nm 女 ” /><Br /> 

<asp:Button ID="Buttonl"” runat="server"” Text=" 添 加 用 户 " onclick= 
"Buttonl Click" /><br /> 


</div> 
</div> 
</form> 


(5) 在 Page_Load 事件 中 ， 编 写 代 码 加 载 用 户 信息 并 显示 。 代 码 如 下 : 


protected void Page Load (object sender, EventArgs e) 


{ 


2 


if(!IsPostBack) 
showUserList(); 


// 从 数据 库 中 查询 所 有 用 户 数据 并 显示 在 页 面 上 


private void showUserList() 


{ 


SqlHelper db = new SqlHelper(); // 创 建 SqlHelper 实例 
// 通 过 SqlHelper 创建 命令 对 象 
DbCommand command = db.GetSqlStringCommond ("Select * from LoginUser") 
// 通 过 SqlHelper 执行 命令 并 获得 数据 读 取 器 
using (DbDataReader reader = db.ExecuteReader (command)) 
{ 
// 将 读 取 到 的 数据 循环 显示 在 页 面 上 
while (reader.Read()) 
{ 
TableRow row = new TableRow(); 
result .Rows.Add (row); 
tor (Lnt = 0 I < oD rl 
{ 
TableCell cell = new TableCell (); 
cell.Text = Convert.ToString (reader[i]); 
row.Cells.Add (cell); 


} 
} 
db.Dispose(); 


(6) 在 “添加 用 户 ” 按 钮 的 Click 事件 中 ， 编 写 代码 完成 添加 用 户 操 作 。 代 码 如 下 : 


protected void Button1l Click(object sender, EventArgs e) 


{ 


-i 


SqlHelper db = new SqlHelper(); // 创 建 SqlHelper 实例 
DbCommand command = db.GetStoredProcCommond ("AddUser"); 
// 得 到 存储 过 程 命令 
db.AddInParameter (command, "Qid"，DbType.String，TextBox1.Text) 
// 向 存储 过 程 添加 参数 
db.AddInParameter (command, "@name", DbType.String, TextBox2.Text); 
db.AddInParameter (command, "@pass", DbType.String, TextBox3.Text); 
db.AddInParameter (command, "@sex",DbType.String, male.Checked ? "M" : 
i 
db.ExecuteNonQuery (command); // 执 行 命令 
db.Dispose(); 
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showUserList (); // 刷 新 用 户 列表 
} 


(7) 运行 SqlHelperDemo.aspx 页 面 ， 运 行 界面 如 图 2.18 所 示 。 


Sqiteiper 应 用 兴 例 - Mo 


BN 


二 c IC 和 


Googe et Wi 
SqlHelper 应 用 沧 讽 中 
用 户 列表 
adnin | 我 的 标题 |343 |2010-2-22 9:07:49 |F| 
luoer01| 王 大 家 1566656 |2008-2-6 0:00:00 a 
user02| 小 红 03 399999 |2008-6-5 0:00:00 |F| 
user03| 小 张 0 |655s8741 |2009-5-2 0:00:00 | 
user04| 新 的 名 称 |adfasdfs| 
user05| 计 算 机 爱好 者 |aasaa H 
user05| 用 户 各 |566656 |2009-5-6 0:00:00 
user07| 小 小 说 h23 |2010-2-21 14:52:43|F 
添加 新 用 户 


用 户 登录 TD，[scmin 


图 2.18 SqlHelper 应 用 示例 


29 小 \ 结 


本 章 介绍 了 .NET 数 据 访问 框架 ADONET 的 相关 知识 ,包含 ADONET 组 成 .ADONET 
包含 的 主要 类 、 使 用 ADONET 访问 数据 库 的 方法 。 使 用 原始 的 ADO.NET 类 和 方法 直接 
操作 数据 库 是 很 烦琐 的 一 个 过 程 ， 本 章 最 后 还 介绍 了 一 个 通用 数据 访问 类 SqlHelper, 封装 
了 常用 ADO.NET 操作 。 使 用 SqlHelper 类 可 以 显著 减少 重复 代码 。 
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相对 于 其 他 Web 开发 技术 来 说 , ASPNET 的 优势 之 一 就 是 简单 高 效 , 能 够 用 更 少 的 代 
码 完成 同样 的 任务 。 数 据 库 操作 是 大 多 数 应 用 程序 的 一 个 核心 功能 ， 在 代码 工作 量 上 占据 
很 大 比重 。ASPNET 对 数据 库 操 作 进行 了 封装 和 简化 ， 提 供 了 一 系列 数据 控件 供 开 发 人 员 
使 用 。ASPNET 数据 控件 可 以 分 为 两 大 类 : 数据 源 控件 和 数据 绑 定 控件 。 数 据 源 控件 是 管 
理 连接 到 数据 源 及 读 取 和 写 入 数据 等 任务 的 ASPNET 控件 。 数 据 绑 定 控件 是 用 来 在 页 面 上 
显示 数据 的 控件 。 本 章 将 介绍 ASPNET 中 的 主要 数据 控件 。 


3.1 ASPNET 数据 绑 定 控件 概述 


数据 绑 定 Web 服务 器 控件 是 指 可 绑 定 到 数据 源 控件 ， 以 实现 在 Web 应 用 程序 中 轻松 
显示 和 修改 数据 的 控件 。 使 用 数据 绑 定 控件 ， 不 仅 能 够 将 控件 绑 定 到 一 个 数据 结果 集 ， 还 
能 够 使 用 模板 自 定义 控件 的 布局 。 同 时 ， 数 据 绑 定 控件 还 提供 用 于 处 理 和 取消 事件 的 方便 


3.1.1 ASP.NET 主要 数据 绑 定 控件 


数据 绑 定 控件 是 用 来 在 页 面 上 显示 数据 的 控件 。 数 据 绑 定 控件 所 显示 的 数据 可 以 来 自 
数据 源 控件 ， 也 可 以 来 自 一 个 集合 对 象 ， 如 DataTable、DataSet， 也 可 以 来 自 Array 或 List。 
将 数据 添加 到 数据 绑 定 控件 以 进行 显示 的 过 程 称 为 数据 绑 定 。 数 据 绑 定 控件 功能 强大 ， 最 
基本 的 功能 就 是 显示 数据 ， 还 可 以 编辑 数据 ， 也 能 够 按照 指定 的 模板 生成 复杂 的 界面 。 
ASPNET 主要 包括 以 下 数据 绑 定 控件 。 

(1) DataList: 可 用 于 显示 任何 重复 结构 中 的 数据 ， 可 按 不 同 的 布局 显示 行 , 例如 按 列 
或 行 对 数据 进行 排序 。 

(2) DetailsView: 可 以 逐一 显示 、 编 辑 、 插 入 或 删除 其 关联 数据 源 中 的 记录 。 即 使 
DetailsView 控件 的 数据 源 公 开 了 多 条 记录 ， 该 控件 每 次 也 只 会 显示 一 条 数据 记录 。 

(3) FormView: 可 以 处 理 数据 源 中 的 单条 记录 ， 该 控件 与 DetailsView 控件 相似 ， 二 
者 区 别 在 于 在 于 DetailsView 控件 使 用 表格 布局 ， 而 FormView 控件 则 不 指定 用 于 显示 记录 
的 预定 义 布局 。 

(4) GridView: 以 表格 的 形式 显示 数据 源 中 的 值 ， 该 表格 中 的 每 一 列 代表 一 个 字段 ， 
每 一 行 代表 一 条 记录 。 使 用 GridView 控件 可 以 选择 和 编辑 这 些 项 ， 也 可 以 进行 排序 。 

(5) Repeater: 是 一 个 数据 绑 定 容 器 控件 ， 用 于 生成 各 个 项 的 列表 。 可 使 用 模板 定义 
网 页 上 各 个 项 的 布局 。 当 网 页 运行 时 ， 该 控件 为 数据 源 中 的 每 一 项 重复 相应 的 布局 。 
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3.1.2 最 简单 的 数据 绑 定 控件 DropDownList 


在 ASPNET 控件 分 类 中 ， 通 常 不 把 DropDownList 分 类 为 数据 绑 定 控件 ， 而 是 分 类 为 
标准 控件 。 但 是 ，DropDownList 控件 确实 具有 数据 绑 定 功能 ， 从 这 个 角度 来 看 ， 和 其 他 的 
数据 绑 定 控件 并 无 区 别 。 由 于 DropDownList 控件 的 简单 性 ， 很 适合 作为 介绍 数据 绑 定 控 
件 的 每 一 个 例子 。 

DropDownList 控件 可 以 绑 定 到 一 个 数据 源 ， 把 数据 源 中 的 数据 显示 在 列表 中 。 在 实际 
项 目 中 ， 经 常用 到 DropDownList 控件 显示 数据 库 中 的 数据 。 例 如 ， 在 很 多 页 面 上 都 可 以 
看 到 让 用 户 从 下 拉 列 表 框 中 选择 省 份 、 选 择 职业 等 ， 如 果 这 个 页 面 是 ASPNET 页 面 ， 那 么 
这 个 控件 通常 就 是 一 个 DropDownList， 绑 定 到 数据 库 中 某 个 表 。 
编程 方式 将 数据 绑 定 控件 绑 定 到 数据 源 ， 通 常 需要 两 个 步骤 。 

(1) 将 数据 绑 定 控件 的 DataSource 属性 设置 为 数据 源 〈 如 一 个 DataTable 的 实例 )。 

(2) 调用 数据 绑 定 控件 的 DataBind0 方 法 ， 执 行 数据 绑 定 。 

数据 绑 定 控件 都 是 直接 或 者 间接 继承 自 BaseDataBoundControl 类 ，DataSource 属性 和 
DataBind 方法 都 在 BaseDataBoundControl 类 中 定义 ， 签 名 如 下 。 

public virtual Object DataSource { get; set; } 

public override void DataBind() 

将 DropDownList 控件 绑 定 到 数据 源 时 ， 还 需要 设置 另外 两 个 属性 : DataValueField 和 
DataTextField 。DataValueField 属性 是 指 列表 项 中 存储 的 值 来 自 于 数据 库 哪 个 字段 ， 
DataTextField 是 指 列表 项 中 所 显示 的 文本 来 自 于 数据 库 哪 个 字段 。 

【 例 3-1】 DropDownList 显示 数据 。 

本 例 从 数据 库 中 读 取 职业 列表 ， 并 在 DropDownList 控件 中 显示 。 


外 提示 : 为 减少 数据 访问 代码 , 节约 篇 幅 , 突出 重点 , 本 例 使 用 了 第 2 章 所 创建 的 SqlHelper 
类 实现 数据 访问 。 本 章 后 面 的 例子 中 也 将 使 用 SqlHelper 类 ， 不 再 一 一 说 明 。 


(1) 创建 一 个 ASPNET Web 应 用 程序 。 

(2) 在 App_Data 文件 夹 下 添加 一 个 数据 库 ， 在 数据 库 中 添加 一 个 职业 表 Occupation， 
创建 表 的 SQL 语句 如 下 : 

create table Occupation (OccupationID int idqentity(1,1) primary key, 

OccupationName nvarchar (20) not null) 

(3) 在 Occupation 表 中 添加 几 条 测试 数据 。 

(4) 在 项 目 中 添加 一 个 页 面 OccupationListaspx， 在 页 面 上 放置 一 个 DropDownList 控 
件 ， 并 且 设 置 控件 的 DataValueField 属性 为 OccupationID 〈 对 应 于 数据 库 Occupation 表 的 
字段 名 称 )， 设 置 控件 的 DataTextField 属性 为 OccupationName。 在 页 面 中 添加 JavaScript 
代码 ， 当 用 户 选择 的 职业 发 生 改变 时 ， 弹 出 提示 框 。 页 面 代码 如 下 : 

<head runat="server"> 


<!-- 下 面 这 段 Javascript 代码 的 功能 为 ， 当 所 选择 的 职业 发 生变 化 时 ， 弹 出 提示 框 --> 
<script type="text/javascript" language="javascript"> 
function showSelection() { 


yy 


人 
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Var select = document .getElementById('list'); 
if (select.selectedIndex > 0) { 
Var option = select.options[select.selectedIndex]; 
var msg = ' 你 选择 的 职业 是 : ' + option.text + '， 职 业 代码 为 : ' + 
option.value; 
alert (msg); 
} 


上 
</script> 
</head> 
<body> 
<form id="forml" runat="server"> 
<div> 
请 选择 一 个 职业 : 
<asp:DropDownList ID="list" runat="server" DataTextField= 
"OccupationName" DataValueField="OccupationID" onchange= 
"showSelection();"> 
</asp:DropDownList><br /> 
</div> 
</form> 
</body> 


(5) 在 页 面 的 Page_ Load 事件 中 ， 编 写 代 码 将 数据 绑 定 到 DropDownList 控件 。 代 码 
如 下 : 


protected void Page Load (object sender, EventArgs e) 
if (!IsPostBack) 

bindOccupationList(); 
: } 


private void bindoccupationList() 
{ 
SqlHelper db=new SqlHelper(); //SqlHelper 为 通用 数据 访问 类 
// 创 建 命令 以 执行 select 语句 检索 数据 
Var command = db.GetSqlstringCommond ("select * from occupation"); 
DataTable table = db.ExecuteDataTable (command); 
// 将 数据 绑 定 到 aropdownlist 控件 
list.DataSource = table; 
list.DataBind(); 
. 


(6) 运行 OccupationList.aspx 页 面 ， 运 行 界面 如 图 3.1 所 示 。 


用 DroplovnList 昱 示 和 至 区 - 
文件 编辑 人 也) 查看 () 历史 @) 书签 @) 工具 XY) 帮助 0 


一 © | t/ahost:1051/0ccupationL TY ~ 


来 自 http 11lecalkestl051 的 页 面 说 : 区 


T A RL 次 伯 工程， 和 RJL 人 到 为: 3 
请 选择 一 个 职业 ， [ 误 件 工程 策 回 


图 3.1 用 DropDownList 绑 定 数据 


“Re 
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3.2 GridView 控件 


GridView 是 所 有 ASPNET hl eat 个 。 功 能 的 强大 通常 与 性 
能 的 开销 是 成 正比 的 ，GridView 控件 的 性 能 相对 来 说 也 要 弱 一 些 。GridView 控件 有 非常 复 
杂 的 属性 、 方 法 、 事 件 列表 ， 有 许多 使 用 技巧 。 本 节 将 介绍 GridView 控件 的 使 用 。 


3.2.1 显示 数据 


数据 绑 定 控件 最 基本 的 功能 就 是 显示 数据 。GridView 控件 默认 情况 下 以 二 维 表格 的 形 
式 显示 数据 。 用 户 可 以 自 定 义 表 格 中 各 个 细节 的 样式 ， 包 括 表 头 、 表 脚 、 每 一 列 、 奇 偶 行 
等 , 从 而 设计 出 美观 的 用 户 界面 。 下 面 先 通过 一 个 例子 来 看 最 基本 的 GridView 控件 的 使 用 。 
【 例 3-2】 简单 GridView 示例 。 
本 例 演示 GridView 控件 的 基本 使 用 , 包括 以 表格 形式 显示 数据 、 设 置 列 标题 、 设 置 样 
式 等 。 
(1) 创建 一 个 ASPNET 应 用 程序 。 
(2) 在 项 目 中 添加 一 个 数据 库 ， 在 数据 库 中 添加 一 个 商品 表 Products， 表 结构 如 下 。 
口 ProductID: nchar(4) 类 型 ， 商 品 编号 ， 主 键 。 
ProductName: nvarchar(20) 类 型 ， 不 为 室 ， 商 品名 称 。 
Price: float 类 型 ， 不 为 空 ， 商 品 价格 。 
Stock: float 类 型 ， 不 为 空 ， 商 品 数量 。 
Description: nvarchar(50) 类 型 ， 可 空 ， 商 品 描 述 。 
(3) 向 表 中 添加 一 些 测 试 数据 。 
(4) 在 项 目 中 添加 一 个 新 页 面 SimpleGridView.aspx。 在 页 面 上 放置 一 个 GridView 控件 ， 
代码 如 下 : 


<asp:GridView ID="GridView]l" runat="server" HorizontalAlign="Center" 
CellPadding="2"> 

</asp:GridView> 

(5) 在 页 面 的 Page_Load 事件 中 编写 代码 将 数据 绑 定 到 GridView 中 。 代 码 如 下 : 


protected void Page Load (object sender, EventArgs e) 


口 
口 
口 
口 


if (!IsPostBack) 
bindGrid(); 
上 
private void bindGrid() 
此 
SqlHelper db = new SqlHelper(); 
DbCommand command = db.GetSqlStringCommond ("select * from products"); 
DataTable table = db.ExecuteDataTable (command); 
GridViewl .DataSource = table; 
GridViewl .DataBind(); 


ls 
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(6) 运行 SimpleGridView.aspx 页 面 ， 运 行 效果 如 图 3.2 所 示 。 

(7) 在 图 3.2 中 ， 数 据 的 列 标题 为 数据 库 的 字段 名 ， 界 面 不 够 友好 。 实 际 应 用 中 ， 要 
使 用 直观 的 中 文 词语 作为 列 标题 。GridView 控件 允许 用 户 自 定义 列 标题 。 在 ASPNET 页 
面 设 计 视图 中 ， 单 击 GridView 控件 右上 角 大 于 号 按钮 ， 则 弹出 GridView 智能 任务 面板 ， 
如 图 3.3 所 示 。 


简单 的 GridYi er 控件 示例 - Wezills Firefex 


文件 旭 ” 闹 辑 外 查看 如 历 史 侣 ) 书签 下) 工具 G) 帮助 如 
De XH Dm Pe 
aa i FE BESET 
iayi ev 控件 示例 


pt select opt 


简单 


ProductID|ProductName|Price|Stock| Description 
0001 噬 140 |10 |86, 白色 

0002 内 存 160 |30 |26 内 存 

0003 显示 器 1258 |6 19 寸 液晶 ， 宽 屏 
0004 显示 器 1350 |2 19 寸 补品 ， 方 屏 | 
0005 噬 75 |e 46， 白 色 / 黑 色 
0006 硬盘 450 ”|10 |5006 PC 硬盘 


EJEd 
图 3.2 GridView 以 表格 形式 显示 数据 图 3.3 ”GridView 智能 任务 面板 


在 图 3.3 所 示 的 GridView 智能 任务 面板 中 ， 选 择 “ 编 辑 列 ”选项 ， 则 弹出 如 图 3.4 所 
示 “ 字 段 ” 对 话 框 。 
在 图 3.4 所 示 对 话 框 中 ,“ 自动 生成 字段 ” 复 选 框 表示 是 否 根 据 数 据 源 自动 生成 列 。 由 
于 接 下 来 要 定制 列 ， 不 需要 自动 生成 列 ， 所 以 取消 选中 这 个 复 选 框 。 然后， 从 “可 用 字段 ” 
列表 中 选中 “BoundField” 项 ， 单 击 “ 添 加 ”按钮 ， 则 在 对 话 框 左下 方 的 “ 选 定 的 字段 ” 
列表 中 会 添加 一 个 新 的 字段 ， 在 对 话 框 右边 会 显示 这 个 字段 的 属性 ， 如 图 3.5 所 示 。 
EF 


DoundFio4 尾 性 他 ) 
国 站 上 


厅 和 动 生 友和 6) 生动 于 = 和 加) 


[Ee | 


4 


图 3.4 GridView 编辑 列 对 话 框 图 3.5 字段 属性 


在 图 3.5 所 示 对 话 框 的 右 侧 属 性 面板 中 ，DataField 属性 表示 要 显示 的 数据 库 字 段 名 ， 
设置 为 ProductID, HeaderText 属性 表示 列 标题 , 设置 为 “商品 编号 ”。 到 此 为 止 为 GridView 
添加 了 一 列 。 按 照 同样 的 步骤 ， 继 续 添 加 其 他 各 列 ， 包 括 ProductName、Price、Stock、 
Description。 注 意 在 添加 Price 时 ， 设 置 字段 的 DataFormatString 属性 为 “{0:C}”， 这 个 属 
性 表示 字段 显示 格式 ， 此 处 以 货币 格式 显示 商品 价格 。 

上 述 在 设计 视图 中 的 操作 会 转换 为 对 页 面 对 应 的 aspx 文件 的 修改 。 如 果 不 采 用 设计 视 
图 , 而 是 直接 修改 代码 文件 ， 也 是 完全 可 以 的 。 前述 编辑 字段 的 操作 对 应 的 GridView 代码 


.114 . 
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如 下 : 


<asp:GridView ID="GridViewl" runat="server" HorizontalAlign="Center" 
CellPadding="2" AutoGenerateColumns="False"> 
<Columns> 


<asp:BoundField DataField="ProductID"” HeaderText=" 商 品 编号 ” /> 
<asp:BoundField DataField="ProductName"” HeaderText=" 商 品名 称 ” /> 
<asp:BoundField DataField="Price" DataFormatSstring="{0:C}" Header- 
Text=" 价 格 ” /> 

<asp:BoundField DataField="Stock" HeaderText=" 库 存 数量 " /> 
<asp:BoundField DataField="Description" HeaderText=" 商 品 描 述 " /> 


</Columns> 

</asp:GridView> 

上 述 代 码 中 ，<asp:BoundField 人 表示 这 是 一 个 数据 绑 定 列 ，DataField 属性 表示 此 列 所 
绑 定 到 的 字段 名 ，HeaderText 属性 表示 该 列 的 标题 DataFormatString 属性 表示 显示 列 时 使 
用 的 格式 字符 串 。 

(8) 再 次 运行 SimpleGridView.aspx 页 面 ， 可 以 看 到 ， 列 标题 都 变 成 了 所 指定 的 中 文 ， 
商品 价格 也 变 成 了 货币 ， 运 行 效果 如 图 3.6 所 示 。 

(9) 为 了 获得 更 加 美观 的 页 面 ， 可 以 使 用 合理 搭配 的 彩色 显示 GridView 中 的 数据 。 
GridView 允许 定义 各 个 行 或 列 的 前 景色 、 背 景色 、 字 体 、 边 框 等 样式 。 同 时 ，GridView 还 
自 带 了 几 种 经 典 样式 以 供用 户 选 择 。 在 图 3.3 所 示 的 GridView 智能 任务 面板 中 ， 选 择 “ 自 
动 套用 格式 ”命令 ， 即 可 弹出 如 图 3.7 所 示 的 “自动 套用 格式 ”对 话 框 。 从 对 话 框 左 侧 选 
择 一 种 样式 ， 从 右 侧 即 可 预览 此 样式 外 观 。 单 击 “ 确 定 ” 按 钮 即 可 将 选中 样式 应 用 于 
GridView 控件 。 


x 


自动 套用 格式 


数 # 坟 数 二 多 局 芍 禾 Ed 
状 s 甸 dd 者 绑 郑 B 终 区 绑 


Googke | 本 与-+> 六 - 国 - 


1 简单 的 Gri ayiev 控 件 示例 = | 四 知 钞 元 暑 id FE 元 绑 
商品 编号 商品 名 称 | 价格 。 库存 数量 | 商品 撕 述 数据 绑 数据 绑 数据 绑 数据 绑 救 据 绑 
0001 唾 ¥140.00 |10 86, 白色 定 定 定 定 定 
0002 _ | 内存 ¥160.00 |30 26 内 存 北 8 甸 数 二 甸 数 # 细 政 所 久 热 后 细 
0003 。 | 显示 器 |¥¥1, 258. 00|6 19 寸 液 蝇 ， 宽 屏 | 和 是 
0004 | 显示 器 | 站 1, 350. 00|2 19 寸 液晶 ， 方 屏 | Ee 总 | | 2 
0005 | 哈 ¥75.00 |6 46， 白 色 /黑色 数据 绑 数据 绑 数据 绑 数据 乡 数据 乡 
0006 ”| 硬盘 。 |¥460.00 |10 5006 PC 硬盘 i 4 5 

原 wm | 


图 3.6 设置 了 列 标题 和 格式 的 GridView 图 3.7 自动 套用 格式 对 话 框 
应 用 了 “传统 型 ”格式 的 GridView 代码 如 下 : 


<asp:GridView ID="GridViewl" runat="server" HorizontalAlign="Center" 
CellPadding="4" AutoGenerateColumns="False" ForeColor="#333333" 
GridLines="None"> 
<Columns> 


= 
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<asp:BoundField DataField="ProductID"” HeaderText=" 商 品 编号 ” /> 
<asp:BoundField DataField="ProductName"” HeaderText=" 商 品名 称 ” /> 
<asp:BoundField DataField="Price" DataFormatstring="{0:C}" 
HeaderText=" 价 格 ” /> 
<asp:BoundField DataField="Stock" HeaderText=" 库 存 数量 "” /> 
<asp:BoundField DataField="Description"” HeaderText=" 商 品 描述 " /> 
</Columns> 
<%--RowStyle 指定 了 行 的 样式 --%> 
<RowStyle BackColor="#EFF3FB" /> 
<%--FooterStyle 指定 了 页 脚 样式 --%> 
<FooterStyle BackColor="#507CD1" Font-Bold="True" ForeColor="White" /> 
<%--PagerStyle 指定 了 分 页 区 域 的 样式 --%> 
<PagerStyle BackColor="#2461BF" ForeColor="White" HorizontalAlign= 
"Center" /> 
<%--SelectedRowStyle 指定 了 选中 行 的 样式 --%> 
<SelectedRowStyle BackColor="#D1DDF1" Font-Bold="True" ForeColor= 
-4333333" /> 
<%--HeaderStyle 指定 了 列 标题 的 样式 --%> 
<HeaderStyle BackColor="#507CD1" Font-Bold="True" ForeColor="White" /> 
<%--EditRowStyle 指定 了 正在 被 编辑 的 行 的 样式 --%> 
<EditRowStyle BackColor="#2461BF" /> 


<%--AlternatingRowStyle 指定 了 偶数 行 的 样式 --%> 
<AlternatingRowStyle BackColor="White" /> 
</asp:GridView> 


3.2.2 数据 排序 


GridView 支持 数据 排序 功能 。 默 认 情 况 下 ， 排 序 功 能 未 启用 ， 可 以 通过 设置 
AllowSorting 属性 为 true 启用 排序 。 如 果 GridView 启用 了 排序 功能 ， 可 排序 的 列 标题 显示 
为 超 链接 ， 单 击 某 一 列 标题 ， 即 可 实现 将 数据 按照 这 一 列 排序 ， 再 单 击 一 次 该 列 标题 ， 则 
切换 排序 方式 ， 升序 变 为 降序 ， 降 序 变 为 升序 。 

【 例 3-3】 GridView 排序 。 

GridView 支持 排序 功能 。 本 例 将 演示 如 何 实现 这 个 功能 。 

(1) 按照 例 3-2 所 述 步骤 创建 一 个 页 面 GridViewSort.aspx， 也 可 以 直接 使 用 例 3-2 所 
创建 的 页 面 SimpleGridView.aspx。 

(2) 将 GridView 控件 的 AllowSorting 属性 设置 为 tue。 

(3) 将 GridView 控件 中 各 个 字段 的 SortExpression 属性 设置 为 各 自 的 字段 名 称 。 设 置 
完成 后 的 GridView 代码 如 下 : 

<asp:GridView ID="GridViewl" runat="server" HorizontalAlign="Center" 

AutoGenerateColumns="False" AllowSorting="True" onsorting= 


"GridView1l Sorting" > 

<Columns> 
<asp:BoundField DataField="ProductID" HeaderText=" 商 品 编号 " SortEx- 
pression="ProductID" /> 
<asp:BoundField DataField="ProductName"” HeaderText=" 商 品名 称 " SortEx- 
pression="ProductName" /> 
<asp:BoundField DataField="Price" DataFormatSstring="{0:C}" HeaderText 
=" 价 格 " SortExpression="Price"/> 


<asp:BoundField DataField="Stock" HeaderText=" 库 存 数 量 " SortExpression 
="Stock" /> 
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<asp:BoundField DataField="Description" HeaderText=" 商 品 描述 " /> 
</Columns> 
</asp:GridView> 


(4) 在 GridView 的 Sorting 事件 中 ， 按 照 所 选择 的 字段 对 数据 进行 排序 ， 重 新 绑 定 
GridView。 代 码 如 下 : 


protected void GridView] Sorting (object sender, GridViewSortEventArgs e) 
| 
bindGrid(e.SortExpression, 
e.SortDirection==SortDirection.Descending); 
} 
/// <summary> 
/// 根据 排序 字段 绑 定 数据 到 GridView 
/// </summary> 
/// <param name="sortField"> 要 排序 的 字段 </param> 
/// <param name="desc"> 是 否 降 序 </param> 
private void bindGrid(string sortField,bool desc) 
{ 
SqlHelper db = new SqlHelper(); 
string sql = "select * from products "; 
// 如 果 排序 字段 不 为 室 ， 则 在 select 语句 后 面 添加 order by 子 句 
if (!string.IsNullOrEmpty (sortField) ) 
{ 
sql += " order by " + sortField; 
// 如 果 desc 参数 为 真 ， 则 在 select 语句 中 order by 列 名 后 面 添加 desc 关键 字 
if (desc) 
SGL 14= " dese "s 


} 


// 执 行 查询 命令 ， 得 到 DataTable， 将 其 绑 定 到 GridView 
DbCommand command = db.GetSqlStringCommond (sql) 7 
DataTable table = db.ExecuteDataTable (command); 
GridViewl .DataSource = table; 
GridViewl .DataBind(); 

} 


(5) 在 Page_Load 事件 中 ， 编 写 代码 加 载 全 部 数据 。 代 码 如 下 : 


protected void Page Load (object sender, EventArgs e) 
四 

if (!IsPostBack) 

{ 


bindGrid(null, false); // 加 载 所 有 数据 ， 不 排序 
} 
外 
(6) 运行 页 面 ， 可 以 看 到 所 有 设置 了 SortEx- 
pression 属性 的 列 标题 都 变 成 了 链接 按钮 ， 单 击 列 标 。 RICE 于 
题 可 以 按照 此 列 进行 排序 。 运 行 界面 如 图 3.8 所 示 。 ew en- 同 


Ceose EE BEE 
| 


aasp://leealkes~avieesert sxpe] ~ 
3.2.3 ”数据 分 页 本 
|o0o3 本 
当 需 要 显示 的 数据 很 多 时 , 如 果 把 数据 全 部 显示 一 售 : 
在 页 面 中 , 则 页 面 会 变 得 很 大 而 且 混 乱 。 一 种 改进 的 ” 压 
做 法 是 将 数据 分 页 显示 ， 每 页 显示 固定 数量 (如 10 图 3.8 GridView 排序 示例 


加 


= 
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条 数据 )， 用 户 通过 翻 页 查看 其 余数 据 。GridView 支持 数据 分 页 功能 ， 可 以 通过 设置 
AllowPaging 属性 为 true 启用 分 页 功能 。 

【 例 3-4】 GridView 分 页 。 

本 例 演示 GridView 如 何 分 页 显示 数据 。 

(1) 创建 一 个 ASPNET 应 用 程序 ， 把 前 面 所 创建 的 包含 Products 表 的 数据 库 复制 到 
App_Data 目录 下 。 

(2) 在 项 目 中 添加 一 个 页 面 GridViewPaging.aspx。 

(3) 在 页 面 上 放置 一 个 GridView, 设置 其 各 个 字段 , 并 将 GridView 控件 的 AllowPaging 
属性 设置 为 true。 

(4) 在 页 面 上 放置 两 个 TextBox 和 两 个 Button, 用 以 设置 GridView 页 面 大 小 和 当前 页 
面 。 所 有 控件 放置 完成 后 GridViewPaging.aspx 页 面 代 码 如 下 : 


<form id="forml" runat="server"> 

<div> 
每 页 记录 数 <asp:TextBox ID="size" runat="server"></asp:TextBox> 
<asp:Button ID="Buttonl" runat="server" Text=" 修 改 " onclick="Buttonl_ 
Click" /><br /> 
转 到 指定 页 <asp:TextBox ID="page" runat="server"></asp:TextBox> 
<asp:Button ID="Button2" runat="server"” Text=" 查 看 " onclick="Button2_ 
Click" /><br /><br /> 

<asp:GridView ID="GridViewl" runat="server" HorizontalAlign= 


"Center" 
AutoGenerateColumns="False" AllowPaging="True" PageSize="5" 
onpageindexchanging="GridViewl PageIndexChanging" > 
<Columns> 


<asp:BoundField DataField="ProductID"” HeaderText=" 商 品 编号 "” /> 
<asp:BoundField DataField="ProductName" HeaderText=" 商 品名 称 ”/> 
<asp:BoundField DataField="Price" DataFormatString="{0:C}" 
HeaderText=" 价 格 " /> 
<asp:BoundField DataField="Stock" HeaderText=" 库 存 数量 " /> 
<asp:BoundField DataField="Description" HeaderText=" 商 品 描述 " /> 
</Columns> 
</asp:GridView> 
</div> 
</form> 


(5) 在 Page Load 事件 中 ， 绑 定数 据 。 代 码 如 下 : 


protected void Page Load (object sender, EventArgs e) 
{ 
if (!IsPostBack) 
{ 
bindGrid(); 
); 
有 
// 查 询 商品 信息 并 绑 定 到 GridView 控件 
private void bindGrid() 
{ 
SqlHelper db = new SqlHelper(); 
DbCommand command = db.GetSqlSstringCommond ("select * from Products"); 
DataTable table = db.ExecuteDataTable (command); 
GridViewl .DataSource = table; 
GridViewl .DataBind(); 


有 
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(6) 在 GridView 的 PageIndexChanging 事件 中 ， 设 置 新 的 页 号 ， 并 重新 绑 定 数据 。 


Protected void GridViewl PageIndexChanging (object sender, 
GridViewPageEventArgs e) 
{ 


GridViewl .PageIndex = e.NewPageIndex; 
bindGrid(); 


(7) 在 设置 页 面 大 小 按钮 的 Click 事件 中 ， 编 写 代码 修改 GridView 页 面 大 小 ， 并 重 


protected void Button1l_Click(object sender, EventArgs e) 
{ 


同 
妾 


int nz 
if (int.TryParse(size.Text, out n)) 
{ 
GridView1.PageSize = n; 
bindGrid(); 
} 


(8) 在 跳 转 页 面 按 钮 Click 事件 中 ， 编 写 代 码 设置 GridView 控件 的 当前 页 面 索引 ， 并 
protected void Button2 Click(object sender, EventArgs e) 


int n; 
if(int.TryParse (page.Text,out n)) 
{ 
//GridView 的 页 号 从 0 开始 ， 而 TextBox 中 输入 的 页 号 从 1 开始 ， 所 以 绑 定时 要 减 1 
GridView1.PageIndex = n-1; 
bindGrid(); 
} 


(9) 运行 页 面 ， 运 行 界面 如 图 3.9 所 示 。 


fox 
文件 四 如 查看 WD 历史 G) 书签 @。 工具 CD) | 帮助 0 


【< © h/t - Ma 
Ce Ee :MR ) 


| Deriayien 分 页 < 


商品 编号 | 商品 名 称 | ”价格 ”| 库存 数量 | 商品 描述 
0001 0 盘 ¥140.00 |10 I86， 白 色 
0002  _ | 内存 ¥160.00 |30 |26 内 存 

0003 | 显示 器 |¥1, 258. 00|6 19 寸 液晶 ， 宽 屏 | 
0004 | 蝎 示 器 |¥1, 350. 00lz 19 寸 液晶 , 方 屏 | 


Eg 
图 3.9 ”GridView 分 页 示例 


例 3-4 利用 了 GridView 自 带 的 分 页 功能 分 页 显示 数据 。 这 种 方式 虽然 简单 ， 却 有 非常 
严重 的 性 能 问题 。 通 过 例子 中 的 bindGrid 代码 可 以 看 出 ,程序 首 先 把 数据 库 中 的 所 有 数据 
读 出 ， 保 存 到 DataTable 中 ， 然 后 再 把 DataTable 绑 定 到 GridView 控件 上 ， 由 GridView 进 
行 分 页 显示 。 如 果 数 据 库 里 数据 量 很 大 , 例如 一 个 网 上 商店 可 能 会 有 上 百 万 甚至 更 多 商品 ， 


“ls 
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那么 把 这 些 数据 全 部 读 出 将 浪费 极 大 的 服务 器 数据 库 和 内 存 资源 ， 大 大 降低 服务 器 性 能 。 

实际 上 , 由 于 用 户 是 一 页 一 页 查看 数据 的 , 程序 没有 必要 把 全 部 数据 都 从 数据 库 读 出 ， 
而 只 需要 读 出 当前 页 需要 显示 的 那 部 分 数据 。 这 种 一 页 一 页 从 数据 库 读 取 数 据 并 且 显 示 的 
分 页 方式 才 是 正确 的 高 性 能 分 页 方式 。 用 这 种 方式 进行 数据 分 页 时 ， 通 常 的 做 法 是 写 一 个 
存储 过 程 读 取 指定 页 面 的 数据 。 

综 上 所 述 , 虽然 GridView 自 带 分 页 功能 , 但 是 这 个 功能 在 实际 项 目 中 应 用 很 少 ,除非 
明确 知道 程序 中 的 数据 量 非常 少 ， 和 否则 不 应 该 使 用 这 种 分 页 方式 。 正 确 的 做 法 应 该 使 用 存 
储 过 程 分 页 。 下 面 通过 一 个 例子 具体 说 明 存 储 过 程 分 页 的 实现 方法 。 

【 例 3-5】 存储 过 程 分 页 。 

本 例 利 用 存储 过 程 进行 数据 分 页 并 显示 在 GridView 中 ， 例 子 中 用 到 一 个 第 三 方 控件 
AspNetPager。 

(1) 创建 一 个 ASPNET Web 应 用 程序 ， 把 例 3-4 所 创建 的 数据 库 复制 到 项 目 中 。 

(2) 创建 分 页 存储 过 程 ， 其 功能 就 是 根据 页 码 和 页 面 大 小 读 取 一 写 的 数据 。 例 如 ， 页 
面 大 小 为 10， 当 前 要 显示 第 3 页 ， 那 么 存储 过 程 应 该 读 取 第 21 到 30 条 数据 。 用 一 个 公式 
来 描述 ， 要 读 取 的 数据 为 : 从 (pageIndex -1) * pageSize+1l 到 pageIndex*pageSize。 在 SQL 
Server 2005 中 ， 有 一 个 系统 函数 Row_Number() 可 以 得 到 一 行 数据 在 整个 记录 和 集中 的 行 数 。 
分 页 读 取 数据 的 存储 过 程 代码 如 下 : 

CREATE PROCEDURE dbo.GetProductByPage 


@pageIndex int, 
@pageSize int 


AS 
declare @min int; // 要 读 取 数 据 的 最 小 行 号 
declare @max int; // 要 读 取 数据 的 最 大 行 号 
// 计 算 @min 和 @max 


set @min=@pageSize*(@pageIndex-1)+1; 

set @max=@pageSize*@pageIndex; 

// 下 面 这 条 较为 复杂 的 sql 语句 的 作用 是 将 小 括号 中 的 select 语句 结果 当 作 一 个 表 
myTable 

// 然 后 再 从 myTable 中 查询 指定 范围 的 数据 

with myTable as 

( 

select ProductID, ProductName, Price, Stock, Description, 

Row Number() over (order by ProductID) as rownum 

from Products 

) 

select ProductID, ProductName, Price, Stock, Description from myTable 

where rownum between min and @max 

RETURN 


全 提示 : Row_Number(O 函 数 是 SQL Server 2005 中 新 增 的 函数 ， 作 用 为 获取 当前 行 在 查询 
结果 集中 的 行 数 . Row_ Number 函数 后 必须 用 over 关键 字 指 定 查 询 的 排序 规则 。 
(3) 在 项 目 中 添加 一 个 页 面 DbPaging.aspx， 在 页 面 上 放置 一 个 GridView 以 及 Label 
和 TextBox， 参 照 以 下 代码 。 
每 页 记录 数 <asp:TextBox ID="size" runat="server"></asp:TextBox> 


<asp:Button ID="Buttonl" runat="SerVer" Text=" 修 改 " onclick="Button1l 
Click" />be > 
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转 到 指定 页 <asp:TextBox ID="pageNum" runat="server"></asp:TextBox> 
<asp:Button ID="Button2" runat="server" Text=" 查 看 " onclick="Button2 
Ceckm /<b /> 
提示 : 共有 <asp:Label ID="count" runat="server" Text=""></asp:Label> 条 数据 
<Br /><br /><bre /> 
<asp:GridView ID="GridViewl" runat="server" HorizontalAlign="Center" 
AutoGenerateColumns="False" > 
<Columns> 
<asp:BoundField DataField="ProductID" HeaderText=" 商 品 编号 " /> 
<asp:BoundField DataField="ProductName"” HeaderText=" 商 品名 称 "” /> 
<asp:BoundField DataField="Price" DataFormatString="{0:C}"” Header 
Text=" 价 格 ” /> 
<asp:BoundField DataField="Stock" HeaderText=" 库 存 数量 " /> 
<asp:BoundField DataField="Description"” HeaderText=" 商 品 描述 " /> 
</Columns> 
</asp:GridView> 


(4) 从 网 上 (http://down.chinaz.com/soft/10039.htm) 下 载 第 三 方 控件 AspNetPager， 并 
添加 到 Visual Studio 工具 箱 中 。 然 后 ， 把 AspNetPager 控件 添加 到 DbPaging.aspx 页 面 。 


<webdiyer:AspNetPager ID="pagerl" runat="server" onpagechanged="pagerl 
PageChanged"> 
</webdiyer:AspNetPager> 


(5) 在 Page_Load 事件 中 ， 编 写 代 码 绑 定数 据 。 


protected void Page Load (object sender, EventArgs e) 
{ 
if (!IsPostBack) 
于 
bindGrid(); 
} 
} 
// 页 面 首次 加 载 时 调用 此 方法 加 载 并 显示 全 部 数据 
private void bindGrid() 
{ 
// 查 询 所 有 商品 信息 并 绑 定 到 GridView 控件 
SqlHelper db = new SqlHelper(); 
DbCommand command = db.GetSqlStringCommond ("select * from products"); 
DataTable table = db.ExecuteDataTable (command); 
GridViewl .DataSource = table; 
GridViewl .DataBind(); 
// 读 取 商 品 总 数 并 设置 分 页 控件 
command = db.GetSqlSstringCommond ("select count (*) from products"); 
int n = Convert.ToInt32 (db.ExecuteScalar (command)); 
count.Text = n.ToString(); 
// 将 分 页 控件 的 总 记录 数 和 页 面 大 小 都 设置 为 查询 到 的 商品 总 数 ， 从 而 在 一 页 显示 全 部 商品 
pagerl .RecordCount = n; 
pagerl .PageSize = n; 


} 
(6) 在 AspNetPager 控件 的 PageChanged 事件 中 , 读 取 一 页 数据 并 显示 在 GridView 中 。 
// 读 取 一 页 数据 并 显示 


private void bindPageData () 


D 


SqlHelper db = new SqlHelper (); 


// 创 建 一 个 命令 以 存储 过 程 
DbCommand command = db.GetStoredProcCommond ("GetProductByPage"); 


= 
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// 向 存储 过 程 添加 两 个 参数 〈 当 前 页 号 和 页 面 大 小 ) 
db.AddInParameter (command, "@pageIndex" ,DbType.Int32 , pagerl. 
CurrentPageIndex); 
db.AddInParameter (command, "@pageSize", DbType.Int32, pagerl. 
PageSize); 
// 执 行 存储 过 程 ， 得 到 数据 ， 绑 定 到 GridqView 
DataTable table = db.ExecuteDataTable (Command) 
GridViewl .DataSource = table; 
GridViewl .DataBind(); 
// 分 页 控件 当前 页 发 生变 化 时 ， 重 新 绑 定 数据 
protected void pagerl PageChanged (object sender, EventArgs e) 
{ 
bindPageData (); 


(7) 在 设置 页 面 大 小 按钮 的 Click 事件 中 ， 更 改 默认 页 面 大 小 。 


protected void Buttonl Click(object sender, EventArgs e) 
{ 
int n; 
if (int.TryParse(size.Text, out n)) 
{ 
pagerl.PageSize = n; 
bindPageData (); 
} 


(8) 在 跳 转 页 面 按钮 的 Click 事件 中 ， 显 示 指 定 页 面 的 数据 。 


protected void Button2 Click(object sender, EventArgs e) 
{ 
int ns; 
if (int.TryParse(size.Text, out n)) 
{ 
pagerl .CurrentPageIndex = n; 
bindPageData (); 


} 
(9) 运行 DbPaging.aspx 页 面 ， 运 行 界面 如 图 3.10 所 示 。 


存 钱 过 程 分 页 - Wozrills Firefox 
文件 中) 编辑 他) 查看 WW) 历史 人 ) 书签 @) 工具 帮助 0) 


全- Ce /oo - 由 3 ae 户 
EL 是 


| 存 鱼 过 程 分 页 =| _ - 
每 页 记录 数 风 修改 
转 到 指定 页 i 查看 
提示 ， 共 有 6 条 数据 
商品 编号 商品 名 称 | ”价格 “| 库存 数量 商品 描述 
0001 噬 | ¥140.00| 10 36, 白色 
0002 | 内 存 | ¥160.00 | 30 26 内 存 
0003 | 显示 器 |¥1, 258.00| ”6 _|19 寸 涪 晶 ,宽屏 | 
0004 | 显示 器 | 站 1, 350.00| ”2 ”|19 寸 液晶 ， 方 屏 


X12 
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图 3.10 存储 过 程 分 页 
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3.2.4 删除 数据 


GridView 是 一 个 功能 强大 的 数据 绑 定 控件 ， 不 仅 能 够 显示 数据 ， 而 且 能 够 删除 和 编辑 
数据 。 当 然 ，GridView 是 一 个 数据 绑 定 控件 ， 仅 提供 数据 显示 和 操作 的 界面 ， 不 执行 对 数 
据 库 的 实际 操作 ， 如 读 取 数据 、 更 新 数据 等 。 用 GridView 编辑 和 删除 数据 时 ， 必 须 配合 代 
码 或 者 数据 绑 定 控件 。 前 面 所 介绍 的 GridView 的 排序 和 分 页 功能 也 是 需要 代码 或 者 数据 源 
控件 的 配合 才能 工作 。 

【 例 3-6】 删除 数据 。 

本 例 演示 如 何在 GridView 中 删除 数据 。 

(1) 创建 一 个 ASPNET 应 用 程序 ， 把 前 面 所 创建 的 包含 Products 表 的 数据 库 复制 到 
App_Data 目录 下 。 

(2) 在 项 目 中 添加 一 个 页 面 GridViewDelete.aspx。 

(3) 在 页 面 上 放置 一 个 GridView， 设 置 其 各 个 字段 以 显示 Products 表 中 商品 信息 。 


<asp:GridView ID="GridViewl" runat="server" HorizontalAlign="Center" 
AutoGenerateColumns="False"> 
<Columns> 


<asp:BoundField DataField="ProductID"” HeaderText=" 商 品 编号 " /> 
<asp:BoundField DataField="ProductName" HeaderText=" 商 品名 称 ” /> 
<asp:BoundField DataField="Price" DataFormatString="{0:C}" Header- 
Text=" 价 格 "” /> 
<asp:BoundField DataField="Stock" HeaderText=" 库 存 数量 " /> 
<asp:BoundField DataField="Description"” HeaderText=" 商 品 描 述 " /> 
</Columns> 
</asp:GridView> 
(4) 在 GridViewDelete.aspx 页 面 的 设计 视图 中 ， 单 击 GridView 控件 右上 角 智 能 按钮 
打开 智能 面板 ,然后 选择 “添加 新 列 ” 选 项 ， 则 弹出 如 图 3.11 所 示 的 “添加 字段 ”对 话 框 。 
在 图 3.11 所 示 “ 添 加 字段 ”对 话 框 中 ， 从 顶部 的 “选择 字段 类 型 ”下 拉 列 表 框 中 选择 
CommandField， 将 页 眉 文本 设置 为 “删除 ”从 “按钮 类 型 ”下 拉 列 表 框 表 中 选择 Link， 
在 命令 按钮 中 选中 “删除 ”。 


全 提示 : GridView 的 字段 有 几 种 类 型 ， 最 基本 的 是 绑 定 字段 BoundField， 其 作用 是 以 纯 文 
本 形式 显示 数据 库 里 的 数据 。 本 例 所 使 用 的 命令 字段 CommandField 在 GridView 
中 显示 为 一 个 包含 命令 按钮 的 列 ， 这 些 按钮 可 以 触发 服务 器 端的 命令 事件 。 
GridView 的 标准 命令 有 4 种 : 编辑 Edit、 删 除 Delete、 更 新 Update 和 选择 Select。 


经 过 如 上 操作 以 后 ， 从 页 面 代码 视图 中 可 以 看 到 GridView 代码 多 出 一 行 。 

<asp:CommandField HeaderText=" 删 除 " ShowDeleteButton="True" ShowHeader= 

"True" /> 

(5) 将 GridView 的 DataKeyNames 属性 设置 为 ProductID。DataKeyNames 属性 表示 
GridView 所 显示 数据 的 主键 字段 ，GridView 的 删除 、 编辑 数据 都 需要 根据 主键 生成 相应 的 
SQL 语句 。 

(6) 在 GridView 的 RowDeleting 事件 中 编写 代码 从 数据 库 删 除数 据 。 
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protected void GridView1 RowDeleting(object sender, GridViewDelete-— 
EventArgs e) 
1 


string key = e-Keys[0] .ToString(): // 得 到 要 删除 行 的 主键 
string sql = "delete from products where ProductID=@id"; 
// 构 建 delete 语句 


SqlHelper db = new SqlHelper(); 

DbCommand command = db.GetSqlStringCommond(sql); 

db.AddInParameter (command，"Q@id"，DbType.String，key) ; // 添 加 参数 

db.ExecuteNonQuery (command); // 执 行 删除 

bindGrid(); // 重 新 绑 定数 据 
} 


(7) 在 页 面 的 Page_Load 事件 中 编写 代码 加 载 数据 。 


protected void Page Load (object sender, EventArgs e) 


{ 
if (!IsPostBack) 
{ 
bindGrid(); 
} 
) 


private void bindGrid() 


1 
// 查 询 商品 信息 并 绑 定 到 GridView 
SqlHelper db = new SqlHelper(); 
DbCommand command = db.GetSqlStringCommond ("select * from products"); 
DataTable table = db.ExecuteDataTable (command); 
GridViewl.DataKeyNames = new string []{"ProductID"};  ”// 设 置 主键 列 名 称 
GridViewl .DataSource = table; 
GridViewl .DataBind(); 

} 


(8) 运行 GridViewDelete.aspx 页 面 ， 运 行 效果 如 图 3.12 所 示 。 
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图 3.11 添加 字段 对 话 框 图 3.12 GridView 删除 数据 
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3.2.5 ”更 新 数据 


在 前 面 的 例子 中 ，GridView 处 于 只 读 状 态 ， 数 据 是 以 静态 文本 的 形式 显示 在 页 面 上 ， 
用 户 不 能 编辑 。 可 以 把 GridView 某 一 行 设置 为 编辑 状态 , 那么 这 一 行 数据 的 各 个 字段 就 分 
别 显示 在 可 编辑 的 控件 中 《〈 默 认为 TextBox)。GridView 的 EditIndex 属性 表示 处 于 编辑 状 
态 的 行 下 标 〔 从 0 开始 )， 如 果 设 置 为 -1， 则 表示 没有 行 处 于 编辑 状态 。 

如 前 所 述 ，GridView 仅仅 是 一 个 显示 和 编辑 数据 的 界面 ， 不 与 数据 库 发 生 联 系 。 要 想 
把 编辑 的 数据 保存 到 数据 库 中, 还 需要 在 适当 的 事件 中 编写 代码 , 完成 实际 的 数据 库 操作 。 
GridView 控件 与 编辑 更 新 数据 相关 的 事件 主要 有 RowEditing、RowCanelingEdit 和 
RowUpdating。 

【 例 3-7】 更 新 数据 。 

本 例 演示 如 何在 GridView 中 编辑 和 保存 数据 。 

(1) 创建 一 个 ASPNET 应 用 程序 ， 把 前 面 所 创建 的 包含 Products 表 的 数据 库 复制 到 
App_Data 目录 下 。 

(2) 在 项 目 中 添加 一 个 页 面 GridViewUpdate.aspx。 

(3) 在 页 面 上 放置 一 个 GridView， 设 置 其 各 个 字段 以 显示 Products 表 中 的 商品 信息 。 
注意 要 把 ProductID 这 一 列 的 ReadOnly 属性 设置 为 tue， 从 而 不 允许 用 户 编辑 这 一 列 。 

<asp:GridView ID="GridViewl" runat="server" HorizontalAlign="Center" 

onrowcancelingedit="GridView]l RowCancelingEdit" 
onrowediting="GridViewl RowEditing" 
onrowupdating="GridView1l RowUpdating" AutoGenerateColumns="False"> 
<Columns> 
<asp:BoundField DataField="ProductID" HeaderText=" 商 品 编号 " ReadOonly 
="true" /> 
<asp:BoundField DataField="ProductName" HeaderText=" 商 品名 称 "” /> 
<asp:BoundField DataField="Price" DataFormatString="{0:C}j" Header- 
Text=" 价 格 "” /> 
<asp:BoundField DataField="Stock" HeaderText=" 库 存 数 量 "” /> 
<asp:BoundField DataField="Description" HeaderText=" 商 品 描述 " /> 


</Columns> 
</asp:GridView> 


(4) 在 GridView 中 添加 一 个 “编辑 ”的 命令 字段 。 


<asp:CommandField HeaderText=" 编 辑 " ShowHeader="True" ShowEditButton= 
"True" EditText=" 编 辑 ” UpdateText=" 保 存 " CancelText=" 取 消 ” /> 


(5) 在 Page Load 事件 中 ， 加 载 并 显示 数据 。 


protected void Page Load (object sender, EventArgs e) 
{ 

if (!IsPostBack) 

四 


} 


bindGrid(); 


} 
private void bindGrid() 


// 查 询 商品 信息 并 绑 定 到 GridView 
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(6) 当 用 户 在 GridView 中 单 击 一 行 的 编辑 按钮 时 , 会 触发 GridView 的 RowEditing 寺 


SqlHelper db = new SqlHelper (); 

DbCommand command = db.GetSqlStringCommond ("select * from products"); 
DataTable table = db.ExecuteDataTable (command); 

GridViewl .DataSource = table; 

GridViewl .DataBind(); 


4 


件 。 在 此 事件 中 编写 代码 ， 根 据 事件 参数 设置 处 于 编辑 状态 的 行 。 


protected void GridView1l RowEditing (object sender, GridViewEditEventArgs 


e) 


: 


$ 


GridViewl .EditIndex = e.NewEditIndex; // 设 置 编辑 行 
bindGrid(); // 重 新 绑 定数 据 


(7) 当 用 户 在 GridView 中 单 击 编辑 行 的 取消 按钮 时 ,会 触发 GridView 的 RowCanceling- 


Edit 事件 。 在 此 事件 中 再 编写 代码 ， 取 家 


x 


当前 行 的 编辑 状态 。 


protected void GridViewl RowCancelingEdit (object sender, GridViewCancel- 
EditEventArgs e) 


i; 


T 


GridViewl .EditIndex = -1; // 取 消 编 辑 行 
bindGrid(); 


(8) 在 用 户 在 GridView 中 单 击 编辑 行 的 保存 按钮 时 , 会 触发 GridView 的 RowUpdating 
事件 。 在 此 事件 中 再 编写 代码 ， 把 用 户 编辑 的 数据 保存 到 数据 库 。 


protected void GridView1l RowUpdating (object sender, GridViewUpdate-— 
EventArgs e) 


{ 


string key = e.Keys[0] .Tostring(); // 获 得 主键 列 
string name = e.NewValues["ProductName"] .ToString (); // 获 得 编辑 后 商品 名 称 
string price = e.NewValues["Price"] .ToString(); // 获 得 编辑 后 商品 价格 
string stock = e.NewValues["Stock"] .ToString() // 获 得 编辑 后 商品 库存 
string description = e.NewValues["Description"] .ToString(); 

// 获 得 编辑 后 商品 描述 

// 构 建 update 语句 
string sql = "update Products set ProductName=@name, Price=@price," 

+" Stock=@stock, Description=@description " 
+ " where ProductID=@id"; 

SqlHelper db = new SqlHelper(); 
DbCommand command = db.GetSqlstringCommond (sql); 
// 向 update 命令 添加 各 个 参数 
db.AddInParameter (command, "@name", DbType.Sstring, name); 
db.AddInParameter (command, "@price", DbType.Double, price); 
db.AddInParameter (command, "@stock", DbType.Double, stock); 
db.AddInParameter (command, "@description", DbType.Sstring, description); 
db.AddInParameter (command, "@id", DbType.String, key); 
db.ExecuteNonQuery (command); 
GridViewl .EditIndex = -1; // 取 消 编 辑 状态 
bindGrid(); // 重 新 绑 定数 据 


(9) 运行 GridViewUpdate.aspx 页 面 ， 运 行 效果 如 图 3.13 所 示 。 
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图 3.13 GridView 更 新 数据 


3.2.6 ” 光 棒 效果 


在 网 页 上 经 常 看 到 这 样 一 种 特效 ， 随 着 鼠标 移动 ， 鼠 标 下 面 的 网 页 内 容 会 高 亮 显示 ， 
这 种 效果 为 网 页 增色 不 少 。 在 以 表格 形式 显示 数据 的 页 面 上 ， 当 鼠标 在 表格 中 移动 时 ， 鼠 
标 移 到 的 整个 一 行 会 高 亮 显 示 ， 鼠 标 移 出 时 恢复 正常 显示 ， 这 种 效果 一 般 称 为 光 棒 。 光 棒 
效果 是 在 浏览 器 端 实现 的 ， 需 要 JavaScript 支持 。 

【 例 3-8】 GridView 实现 光 棒 效果 。 

本 例 结合 GridView 控件 和 JavaScript 脚本 实现 光 棒 效果 。 

(1) 创建 一 个 ASPNET 应 用 程序 ， 把 前 面 所 创建 的 包含 Products 表 的 数据 库 复 制 到 
App_Data 目录 下 。 

(2) 在 项 目 中 添加 一 个 页 面 LightBeam.aspx。 

(3) 在 页 面 上 放置 一 个 GridView， 设 置 其 各 个 字段 以 显示 Products 表 中 商品 信息 。 为 
了 使 页 面 效 果 更 加 明显 ， 设 置 GridView 的 表 头 、 奇 偶 行 背景 色 不 一 致 。 


<asp:GridView ID="GridViewl" runat="server" HorizontalAlign="Center" 
AutoGenerateColumns="False" onrowcreated="GridView1l RowCreated" 
CellPadding="4" GridLines="None" > 
<Columns> 
<asp:BoundField DataField="ProductID" HeaderText=" 商 品 编号 " /> 
<asp:BoundField DataField="ProductName" HeaderText=" 商 品名 称 " /> 
<asp:BoundField DataField="Price" DataFormatString="{0:C}" 
HeaderText=" 价 格 "” /> 
<asp:BoundField DataField="Stock" HeaderText=" 库 存 数量 " /> 
<asp:BoundField DataField="Description"” HeaderText=" 商 品 描述 " /> 
</Columns> 
<HeaderStyle BackColor="#0099FF" /> 
<RowStyle BackColor="#EFF3FB" /> 
<AlternatingRowStyle BackColor="White" /> 
</asp:GridView> 


(4) 在 Page_Load 事件 中 加 载 数 据 并 绑 定 到 GridView。 


protected void Page Load (object sender, EventArgs e) 


机 
if (!IsPostBack) 


{ 
// 查 询 所 有 商品 信息 并 绑 定 到 GridView 
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SqlHelper db = new SqlHelper(): 

DbCommand command = db.GetSqlStringCommond ("select * from 
products"); 

DataTable table = db.ExecuteDataTable (command); 
GridView1.DataSource = table; 

GridView]l .DataBind(); 


(5) 在 LightBeam.aspx 页 面 的 head 标记 中 编写 两 个 JavaScript 函数 ， 实 现 鼠 标 进 入 和 
离开 GridView 某 一 行 时 改变 和 恢复 背景 色 。 鼠标 进入 时 , 把 当前 行 的 原 有 背景 色 保存 到 变 
量 originalColor 中 ， 然 后 将 当前 行 背景 色 设置 为 浅黄 色 ， 当 鼠标 离开 时 ， 将 当前 行 背景 色 
恢复 成 原来 的 背景 色 ， 即 变量 originalColor 所 保存 的 值 。 

<head runat="server"> 


<title>GriqdView 光 棒 效果 </title> 
<script type="text/javascript" language="javascript"> 


// 鼠 标 进行 某 行 时 调用 此 函数 高 亮 显示 该 行 


// 参 数 e 表示 鼠标 移 到 的 行 

function mouseenter (e) { 
originalColor = e.style.backgroundColor; // 得 到 原 有 背景 色 
e.style.backgroundColor = '#ffffdd'; // 设 置 新 背景 色 

} 


// 鼠 标 移出 某 行 时 调用 此 函数 恢复 默认 样式 
function mouseleave (e) { 
e.style.backgroundColor = originalColor; // 恢 复 成 原来 的 背景 色 
} 
</script> 
</head> 


(6) 在 GridView 控件 的 RowCreated 事件 中 ， 编 写 代 码 ， 将 每 一 行 的 鼠标 进入 和 离开 
事件 关联 到 刚才 所 写 的 JavaScript 函数 。 


protected void GridView1 RowCreated (object sender, GridViewRowEventArgs e) 
{ 
if (e.Row.RowType != DataControlRowType.DataRow) 
return; 
// 调 用 mouseenter () 和 mouseover() 函数 ， 把 当前 行 作为 参数 传递 
e.Row.Attributes.Add ("onmouseover", "mouseenter (this);"); 
e.Row.Attributes.Add ("onmouseout", "mouseleave (this);"); 


} 
(7) 运行 LightBeam.aspx 页 面 ， 运 行 效果 如 图 3.14 所 示 。 
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图 3.14 GridView 光 棒 效果 
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在 例 3-8 中 ， 为 页 面 上 的 一 个 GridView 控件 实现 了 光 棒 效果 。 如 果 ASPNET 项 目 中 
有 许多 页 面 的 GridView 都 需要 实现 光 棒 效果 ， 按 照例 3-8 的 方法 ， 就 需要 为 每 个 页 面 和 每 
个 GridView 都 重复 相同 的 工作 : 在 页 面 上 添加 两 个 JavaScript 函数 ， 在 GridView 控件 的 
RowCreated 事件 中 为 每 一 行 的 鼠标 事件 注册 JavaScript 函数 。 这 种 方法 显然 不 够 优秀 。 对 
于 这 个 问题 有 一 个 更 好 的 解决 方案 , 就 是 编写 一 个 实现 了 光 棒 效果 的 GridView 控件 , 然后 
在 需要 的 页 面 上 直接 使 用 这 个 控件 即 可 ， 页 面 中 不 需要 写 任 何 代 码 。 

【 例 3-9】 自 定义 GridView 控件 。 

本 例 设计 了 一 个 自 定义 GridView 控件 实现 光 棒 效果 。 

(1) 在 Visual Studio 中 新 建 一 个 ASPNET 服务 器 控件 项 目 MyControls。 

(2) 在 MyControls 项 目 中 添加 一 个 类 MyGridView， 此 类 从 GridView 中 继承 。 

[ToolboxData("<{0}:MyGridView runat=\"server")] 

public class MyGridView:GridView { } 

(3) 在 MyGridView 类 中 定义 两 个 属性 ， 分 别 表示 鼠标 划 过 时 的 背景 色 和 前 景色 〈 文 
字 颜 色 )。 


public Color hoverBackColor { get; set; } 
public Color hoverTextColor { get; set; } 


(4) 在 MyControls 项 目 中 添加 一 个 JavaScript 文件 gridjs， 在 其 中 编写 鼠标 进入 和 移 
出 的 函数 。 

// 鼠 标 移 进 某 行 时 调用 此 函数 ， 设 置 行 的 背景 和 前 景色 

// 参 数 : row 当前 行 ，bgColor 背景 色 ，textColor 前 景色 

function MouseEnter (row, bgColor, textColor) { 


row.style.backgroundColor = bgColor; 
row.style.color = textColor; 


» 鼠标 移出 时 调用 此 函数 ， 人 


function MouseLeave (row) 解决 方案 次 源 管理 器 | 
row.style. es = | 半 
row.getAttribute ("OriginalColor"); 四 reb confie 回 
row.style.color 三 日 团 wycontrols | 
row.getAttribute ("OriginalTextColor"); | 3 攻 
3] id is 总 


涯 tyGriayiew cs 


(5) 在 解决 方案 资源 管理 器 中 选中 gridjs 文件 ， 在 
属性 窗口 中 ， 将 其 生成 操作 属性 设置 为 “ 获 入 的 资源 ”， | EE 
如 图 3.15 所 示 。 通 过 把 gridjs 设置 为 嵌入 的 资源 ,在 生 
成 项 目 时 ， 会 把 gridjs 也 包含 在 生成 的 程序 集中 。 

(6) 在 MyControls 项 目 Properties 文件 夹 下 ， 打 开 
AssemblyInfo.cs 文件 ， 在 其 中 添加 如 下 代码 : 

// 注 册 gird.js 为 javascript 资源 


[assembly:System.Web.UI.WebResource ("MyCon 


trols.grid.js", "text/javascript")] DesienDatelithhesienTineCreatableTypes 
EntityDeploy 


(7) 在 MyGridView 类 中 , 重 写 OnPreRender() 方 法 ， 
刚才 所 定义 的 JavaScript 资源 。 图 3.15 将 文件 设置 为 嵌入 的 资源 


引 
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protected override void OnPreRender (EventArgs e) 

{ 
base.OnPreRender (e); 
string resourceName = "MyControls.grid.js"; 
ClientScriptManager cs = this.Page.ClientScript; 
cs.RegisterClientScriptResource (typeof (MyControls.MyGridView), 
resourceName); 


} 
(8) 在 MyGridView 类 中 ， 重 写 OnRowCreated() 方 法 ， 注 册 浏 览 器 端 鼠 标 移入 和 移出 
事件 ， 调 用 相应 的 JavaScript 函数 。 


protected override void OnRowCreated (GridViewRowEventArgs e) 


{ 


base.OnRowCreated (e); 

GridViewRow row = e.Row; 

// 如 果 当 前 被 创建 的 行 不 是 数据 行 则 不 需要 处 理 ， 函 数 返回 

if (row.RowType != DataControlRowType.DataRow) 
return; 

Color backColor = Color.Empty; 

Color textColor = Color.Empty; 

// 根 据 是 奇偶 行 得 到 原来 的 背景 色 和 文字 色 

if (row.RowIndex $ 2 == 0) 

{ 
backColor = (this.RowStyle.BackColor == Color .Empty) ? Color.White : 
this.RowStyle.BackColor; 
textColor = (this.RowStyle.ForeColor == Color .Empty) ? Color.Black : 
this.RowStyle.ForeColor; 

} 

else 

i 
backColor = (this.AlternatingRowStyle.BackColor == Color.Empty) ? 
Color.White : this.AlternatingRowStyle.BackColor; 
textColor = (this.AlternatingRowStyle.ForeColor == Color.Empty) ? 
Color.Black : this.AlternatingRowStyle.ForeColor; 

} 

// 注 册 鼠 标 移动 事件 

row.Attributes.Add ("onmouseover", String.Format ("MouseEnter (this, 

CO 
ColorTranslator .ToHtml (this.hoverBackColor), 
ColorTranslator.ToHtml (this.hoverTextColor))); 

row.Attributes.Add ("onmouseout", "MouseLeave (this)"); 

// 将 数据 行 原 有 颜色 作为 参数 添 添加 到 行 中 ， 以 供 javascript 使 用 

row.Attributes.Add ("OriginalColor", ColorTranslator.ToHtml 

(backColor) ); 

row.Attributes.Add ("OriginalTextColor", ColorTranslator.ToHtml 

(textColor) ); 

] 


(9) 创建 一 个 ASPNET 项 目 以 测试 MyGridView 自 定义 控件 。 新 建 一 个 ASPNET 项 
目 ， 在 项 目 中 添加 一 个 页 面 MyGridTestPage.aspx， 在 页 面 上 放置 MyGridView 控件 ， 设 置 
控件 的 hoverTextColor 和 hoverBackColor 属性 。 


全 提示 :如 果 Visual Studio 工具 箱 中 没有 MyGridView 自 定义 控件 , 则 重新 生成 MyControls 
项 目 ，MyGridView 控件 就 会 出 现在 工具 箱 中 。 


<#sQ@ Register assembly="MyControls" namespace="MyControls" tagprefix="cc2" 
各 > 


-De 
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<cc2:MyGridView ID="GridView1"” runat="server" hoverBackColor="#ffffcc" 

hoverTextColor="Red" 

AutoGenerateColumns="false" > 

<Columns> 

<asp:BoundField DataField="ProductID"” HeaderText=" 商 品 编号" /> 
<asp:BoundField DataField="ProductName" HeaderText=" 商 品名 称 " /> 
<asp:BoundField DataField="Price" DataFormatString="{0:C}" Header-— 
Text=" 价 格 " /> 
<asp:BoundField DataField="Stock" HeaderText=" 库 存 数量 " /> 
<asp:BoundField DataField="Description"” HeaderText=" 商 品 描述 "” /> 


</Columns> 


<RowStyle BackColor="#EFF3FB" /> 
<AlternatingRowStyle BackColor="White" /> 


</cc2:MyGridView> 


(10) 在 Page_Load 事件 中 将 数据 绑 定 到 MyGridView。 


protected void Page Load (obje' 
{ 

if (!IsPostBack) 

i 


ct sender, EventArgs e) 


SqlHelper db = new SqlHelper(); 
DbCommand command = db.GetSqlStringCommond ("select * from 


products"); 


DataTable table = db.ExecuteDataTable (command); 


GridView1.DataSource = 
GridView]l .DataBind(); 


1 
(11) 运行 页 面 ， 运 行 界面 见 图 3.1 


虽然 例 3-9 和 3-8 运行 界面 看 起 来 很 相似 ， 
但 二 者 实现 思路 完全 不 同 。 例 3-9 的 自 定义 控件 


是 可 复 用 的 ， 任 何 页 面 上 只 要 放 


MyGridView 控件 ， 不 需要 编写 代码 就 可 以 实现 区 
光 棒 效果 。 另 外 ， 二 者 所 产生 的 客户 端 HIML lo003 | 最 示 器 |¥1, 258. ools 19 十 液晶 ， 宽 屏 | 


代码 也 不 一 样 。 
在 MyGridView 自 定义 控件 中 ， 


gridjjs 文件 作为 资源 。 在 使 用 MyGridView 的 页 


table; 


6。 


置 王 举 


0004 | 显示 器 “|¥1, 288. 00l2 19 寸 液晶 ， 方 屏 
0005 上 0 盘 ¥75.00 |6 46， 白 色 和 黑色 
注册 了 lbo06  _ 噶 盘 ¥460.00 J10 5006 PC 硬盘 


E23 Fa 


面 上 ， 会 自动 添加 一 条 引用 JavaScript 文件 的 语 ”图 3.16 自 定义 GridView 控件 实现 光 棒 效 果 
句 。 例 如 ，MyGridTestPage.aspx 页 面 所 生成 的 


HTML 代码 中 包含 以 下 语句 。 


<script 


src="/WebResource.axd?d=9zpWT3uCqAMxPp6-4mxKUXelotMVa78tSIYrE443z7Ml&am 
p;t=634026979432812500" type="text/javascript"></script> 


3.2.7 ”数据 汇总 


当 用 GridView 显示 报表 性 质 的 数据 (如 销售 记录 、 考 试 成 绩 等 ) 时 ,通常 需要 在 表格 
底部 显示 汇总 信息 , 如 总 销售 量 、 总 销售 额 、 平 均 成 绩 等 。 另外, 用 户 经 常 需要 将 GridView 


中 的 数据 导出 至 Excel， 以 方便 后 续 使 月 


上 有。 
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【 例 3-10】 数据 汇总 。 

本 例 演 示 GridView 的 数据 汇总 和 数据 导出 。 

(1) 创建 一 个 ASPNET 应 用 程序 ， 把 前 面 所 创建 的 包含 Products 表 的 数据 库 复制 到 
App _Data 目录 下 。 

(2) 在 项 目 中 添加 一 个 页 面 GridViewSummary.aspx。 

(3) 在 页 面 上 放置 一 个 GridView， 设 置 其 各 个 字段 以 显示 Products 表 中 的 商品 信息 。 
由 于 要 在 GridView 的 Footer 中 显示 汇总 数据 ， 所 以 将 其 ShowFooter 属性 设置 为 tue。 在 
页 面 上 放置 一 个 Button， 用 以 导出 数据 到 Excel。 


<asp:Button runat="server" Text=" 导 出 Excel" id="buttonl" onclick= 

"buttonl1 Click" /><br /> 

<asp:GridView ID="grid" runat="server" HorizontalAlign="Center" AutoGene- 

rateColumns="False" showFooter="true" 

onrowdatabound="grid RowDataBound" > 

<Columns> 
<asp:BoundField DataField="ProductID"” HeaderText=" 商 品 编号 " /> 
<asp:BoundField DataField="ProductName" HeaderText=" 商 品名 称 "” /> 
<asp:BoundField DataField="Price" HeaderText=" 价 格 " /> 
<asp:BoundField DataField="Stock" HeaderText=" 库 存 数量 " /> 
<asp:BoundField DataField="Description" HeaderText=" 商 品 描述 " /> 

</Columns> 

<HeaderStyle BackColor="#0099FF" /> 

<FooterStyle BackColor="#EFF3FB" /> 

</asp:GridView> 


(4) 在 GridView 的 RowDataBound 事件 中 , 逐 行 累加 计算 商品 总 金额 ,并 在 GridView 
的 Footer 中 显示 汇总 数据 。 


Private double totalStock=0; // 总 库存 
Private double totalMoney=0; // 总 金额 
protected void grid RowDataBound (object sender, GridViewRowEventArgs e) 


{ 

// 如 果 当 前 行为 数据 行 ， 逐 行 累 加 各 行 数据 

if (e.Row.RowType == DataControlRowType.DataRow) 

{ 
double stock Convert .ToDouble(e.Row.Cells[3] .Text); 
double price Convert .ToDouble (e.Row.Cells[2] .Text); 
totalStock += stock; 
totalMoney += stock * price; 


| 

// 如 果 当 前 行为 Footer， 则 在 Footer 中 显示 汇总 数据 

else if (e.Row.RowType == DataControlRowType.Footer) 
e.Row.Cells[0] .Text 
e.Row.Cells[1] .Text 
e.Row.Cells[2] .Text 
e.Row.Cells[3] .Text 


"商品 总 库存 "; 
totalStock.ToString(); 
"商品 总 金额 "; 
totalMoney.ToString(); 


Ll 


(5) 在 “导出 ”按钮 的 Click 事件 中 编写 代码 导出 数据 。 


protected void button1l Click(object sender, EventArgs e) 
{ 


Response.Clear (); 
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Response.AddHeader ("content-disposition"，"attachment; filename= 商 品 信 
息 -X1S")3 

Response.Charset = ""; 

Response.ContentType = "application/excel"; // 内 容 类 型 为 Excel 
System.IO.StringWriter stringWrite = new System.I0.SstringWriter(); 
System.Web.UI.HtmlTextWriter htmlWrite = new HtmlTextWriter 
(stringWrite); 

grid.RenderControl (htmlWrite); 

Response.Write (stringWrite.ToString()); 

Response.End(); 


(6) 在 GridViewSummary.aspx 页 面 中 重 写 VerifyRenderingInServerForm() 方 法 ， 取 消 
对 GridView 控件 的 验证 。 和 否则 程序 运行 时 会 出 现 以 下 错误 提示 “类 型 GridView 的 控件 必 
须 放 在 具有 runat=server 的 窗 体 标记 内 ”。 


public override void VerifyRenderingInServerForm(Control control) 


{ 


h 


i (control = grid) return? // 不 验证 GridView 控件 
base.VerifyRenderingInServerForm(control); 


(7) 运行 GridViewSummary aspx 页 面 ， 运 行 效果 如 图 3.17 所 示 ， 导 出 的 Excel 文件 如 
图 3.18 所 示 。 


国 | 加 了 -Is 
文件 全) 编辑 (E) 查看 (Y) 历史 (S) ”书签 (B) 工具 (T) 秋 肋 (H) | x# | 开始 | 插入 页面 布局 公式 数据 证 网 视图 
Ee Tn/ocnos 3 -Tc 向 起 | 
Ca ae sO | es 
Downmars ES 日 全 | 于 工 工 -| 田 "| 血 - 全 -| 于 -| 严 亚 玫 | 才 
| Ld 人 
G12 ” 天 
0001 唾 140 10 86， 白色 | | Bn | 
0o02 ”内存 “|160 “80 | 内存 商品 编号 | 商品 名 称 | 价格。 | 库存 数 生 | 商品 给 述 | 
0005 | 显示 器 |1258 6 19 寸 液晶 ， 宽 屏 2 十 哈 140| 10|86， 白色 
0004 | 显示 器 “|1288 加 19 寸 渡 品 ,， 方 屏 3 2| 内 存 160 30|26 内 存 
0005 唾 加 6 146， 白 色 和 黑色 | 4 3 显示 器 1258| |19 十 液晶， 宽屏 
0006 硬盘 |46o 10 5006 PC 硬盘 5 4 显示 器 1288| 2|19 十 液晶， 方 屏 
商品 总 库存 |64 商品 总 金额 |21374 6 5lo 和 a 75| 46， 白 色 和 黑色 
hE 6 硬盘 460| 10|5006 PC 神 盘 
Ea [Ea 8 商品 总 库存 | 商品 总 金额 | 。 21374 
图 3.17 GridView 数据 汇总 和 导出 图 3.18 导出 的 Excel 数据 


3.3 ”DataList 控件 


DataList 控件 是 一 种 很 常用 的 数据 绑 定 控件 ， 可 以 用 自 定 义 格式 显示 数据 。DataList 
控件 绑 定 到 一 个 数据 源 ， 数 据 源 中 每 一 条 数据 在 DataList 中 称 为 一 项 。DataList 使 用 模板 
来 定义 数据 显示 格式 。DataList 可 以 为 项 、 交 蔡 项 、 选 定 项 和 编辑 项 创建 模板 ， 也 可 以 使 
用 标题 、 脚 注 和 分 隔 符 模 板 自 定 义 DataList 的 整体 外 观 。 


3.3.1 


以 表格 形式 显示 数据 


DataList 控件 最 基本 的 功能 是 以 纯 文 本 的 表格 形式 显示 数据 。 这 一 功能 与 GridView 类 


= Ls 
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似 ， 但 从 开发 人 员 角 度 来 看 ， 二 者 的 实现 方法 不 同 。 在 GridView 中 显示 数据 时 ， 需 要 向 
GridView 中 添加 各 个 绑 定 字段 ， 而 用 DataList 显示 数据 时 ， 需 要 定义 项 模板 。 

DataList 顾名思义 ， 是 一 个 数据 列表 ， 可 以 显示 多 条 数据 。 每 一 条 数据 称 为 DataList 
的 一 项 ， 如 何在 控件 中 显示 一 项 称 为 项 模板 。 例 如 ， 如 果 要 显示 性 别 ， 是 用 Label 显示 ， 
还 是 用 TextBox 显示 ， 还 是 用 RadioButton 显示 ， 这 就 是 项 模板 要 解决 的 问题 。 当 然 ， 项 
模板 的 作用 绝 不 仅 是 用 不 同类 型 的 控件 显示 数据 ， 还 包括 控件 布局 等 强大 功能 。 

【 例 3-11】 DataList 以 表格 形式 显示 数据 。 

本 例 演 示 DataList 以 表格 形式 显示 商品 信息 。 要 注意 体会 例子 中 项 模板 的 定义 和 使 用 。 

(1) 新 建 一 个 ASPNET Web 应 用 程序 , 将 前 面 所 创建 的 包含 Products 表 的 数据 库 复制 
到 项 目 中 App_Data 文件 夹 下 。 

(2) 在 页 面 上 放置 一 个 DataList 控件 。 此 时 页 面 代码 如 下 : 

<asp:DataList ID="DataList1"” runat="server"> </asp:DataList> 

(3) 在 页 面 设计 视图 中 ， 单 击 DataList 控件 右上 角 的 智能 按钮 ， 打 开 智 能 任务 面板 ， 
从 中 选择 “编辑 模板 ”选项 ， 如 图 3.18 所 示 。 

选择 了 “编辑 模板 ”选项 以 后 ，DataList 控件 从 图 3.19 所 示 的 一 个 灰色 固定 只 读 变 成 


A ET Je 
DataListl - 项 模板 


TteaTenplate 


图 3.19 ”DataList 智能 任务 面板 图 3.20 ”DataList 编辑 模板 视图 


当 DataList 处 于 编辑 模板 状态 时 ， 从 外 观 上 看 起 来 就 好 像 一 个 控件 容器 ， 可 以 从 工具 
箱 拖 放 控件 放 到 这 里 。 这 些 控件 及 其 布局 就 是 DataList 的 项 模板 ，DataList 数据 源 中 的 每 
一 条 数据 都 将 使 用 一 组 这 些 控件 来 显示 。 在 DataList 智能 任务 页 面 有 一 个 下 拉 列 表 ， 里 面 
包含 了 DataList 的 所 有 模板 ， 可 以 从 其 中 选择 一 种 模板 进行 编辑 。 图 3.21 所 示 的 DataList 
控件 项 目 模板 中 放置 了 4 个 Label 控件 。 

在 DataList 控件 的 智能 任务 面板 中 选择 “结束 模板 编辑 ”选项 ,以 返回 默认 的 DataList 
设计 界面 ， 这 时 的 界面 就 是 页 面 运行 时 DataList 的 界面 ， 只 是 其 中 的 数据 都 是 虚拟 的 。 可 
以 看 到 , DataList 现在 变 成 了 具有 多 行 的 一 个 Table, 每 一 行 中 都 有 4 个 Label, 这 4 个 Label 
就 是 项 模板 中 所 定义 的 内 容 ， 如 图 3.22 所 示 。 


ee 
Label Label Label Label 


AlternatineltenTenplate 

SelectedltenTenplate 

EditIteaTeaplate 
页 所 和 页 | 

HeaderTenplate 


了 ooterTenplate 
板 


图 3.21 包含 了 控件 的 DataList 项 模板 图 3.22 项 模板 为 4 个 Label 的 DataList 外 观 
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此 时 切换 到 页 面 的 代码 视图 ，DataList 控件 代码 变 成 如 下 内 容 。 


<asp:DataList ID="DataList1” runat="server"> 
<ItemTemplate> 
<asp:Label ID="Labell" runat="server" Text="Label"></asp:Label> 
&nbsp; 
<asp:Label ID="Label2" runat="server" Text="Label"></asp:Label> 
&nbsp; 
<asp:Label ID="Label3" runat="server" Text="Label"></asp:Label> 
&nbsp; 
<asp:Label ID="Label4" runat="server" Text="Label"></asp:Label> 
</ItemTemplate> 
</asp:DataList> 


(4) 到 目前 为 止 ， 已 经 基本 定义 好 了 用 来 显示 数据 的 控件 ， 下 一 步 需要 将 控件 和 要 显 
示 的 数据 联系 起 来 ， 即 将 数据 绑 定 到 DataList 模板 中 的 控件 。 本 例 中 ， 需 要 把 商品 信息 各 
个 字段 分 别 绑 定 到 各 个 Label 上 。 操 作 方 法 是 切换 到 DataList 的 编辑 项 模板 视图 ， 选 中 项 
模板 中 的 一 个 控件 ， 如 第 一 个 Label， 然 后 打开 Label 的 智能 任务 面板 ， 选 择 “编辑 
DataBindings” 命 令 ， 即 弹出 Label 的 数据 绑 定 对 话 框 ， 如 图 3.23 所 示 。 


TSP TACO CE 
JataListl - 项 模板 
Lasp: Label#Labell | 选择 要 晨 定 到 | 的 属性 ， 然 后 可 通过 选择 字段 来 绑 定 它 。 也 可 使 用 自 定义 代码 表达 式 蝴 定 它 , 


编辑 DataBindings- 可 纤 定 展 性 到 ): 


厂 显示 所 有 展 性 内) 


图 3.23 Label 数据 绑 定 对 话 框 


在 图 3.21 所 示 对 话 框 中 ， 从 左 侧 下 拉 列 表 框 中 选择 Text 属性 ， 然 后 选中 右 侧 的 “ 自 
定义 绑 定 ”RadioButton， 在 “代码 表达 式 ” 中 填写 “Eval("ProductName")”( 不 包括 最 外 层 
的 中 文 引 号 )。 其 中 Eval 是 数据 绑 定 控件 的 一 个 方法 ,作用 是 求 方法 参数 所 指定 字段 的 值 ， 
ProductName 为 要 绑 定 的 字段 名 。 这 个 操作 的 作用 是 用 Label 的 Text 属性 设置 为 数据 源 中 
ProductName 字段 的 值 ， 即 把 ProductName 字段 的 值 显示 在 Label 上 。 

按照 同样 的 操作 步骤 ， 为 后 面 3 个 Label 的 Text 属性 设 定数 据 绑 定 表达 式 ， 分 别 绑 定 
到 Price、Stock 和 Description 字段 。 所 有 字段 绑 定 结束 后 ，DataList 代码 成 为 以 下 内 容 。 


<asp:DataList ID="DataList1"” runat="server"> 
<ItemTemplate> 

<asp:Label ID="Labell" runat="server" Text="'<%# Eval ("ProductName") 
%>'></asp:Label> 
gnbsp; 
<asp:Label ID="Label2" runat="server" Text="'<%# Eval ("Price")%®>'> 
</asp:Label> gnbsp; 
<asp:Label ID="Label3" runat="server" Text="'<%# Eval ("Stock")%®>'> 
</asp:Label> gnbsp; 
<asp:Label ID="Label4" runat="server" Text="'<$%# Eval ("Description")®% 
>'></asp:Label> 
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</ItemTemplate> 
</asp:DataList> 


(5) 在 PageLoad 事件 中 ， 编 写 代 码 从 数据 库 读 取 数据 并 且 绑 定 到 DataList 控件 。 


protected void Page Load (object sender，EventRrgs e) 
{ 
if (!IsPostBack) 
SqlHelper db = new SqlHelper(); 
DbCommand command = db.GetSqlStringCommond ("select * from 
products"); 
DataTable table = db.ExecuteDataTable (command); 
DataList1.DataSource = table; 
DataList]1 .DataBind(); 


(6) 运行 SimpleDataList.aspx 页 面 ， 运 行 界面 如 图 3.24 所 示 。 


pr 
(< PO ese 
EE 
spz | 二 - 


只 140 10 86, 白色 

内 存 ”160 30 26 内 存 

| 显示 器 ”1258 ”6 ”19 寸 液 遇 ， 宽屏 
| 显示 器 ”1288 ”2 19 寸 液晶 ， 方 屏 
只 ”75 6 4, 白色 和 黑色 


| 硬盘 ”460 ”10 ”5006 PC 硬盘 


图 3.24 DataList 以 表格 形式 显示 数据 
在 例 3-11 中 , DataList 显示 出 来 的 数据 格式 比较 凌乱 , 各 行 的 同一 字段 没有 垂直 对 齐 。 
出 现 这 种 情况 的 原因 可 以 从 页 面 生 成 的 HTML 源码 中 找到 。SimpleDataListaspx 生成 的 
HTML 源码 关键 代码 如 下 : 


<table id="DataListl" cellspacing="0" align="Center" rules="all" border 
="1" style="border-collapse:collapse;"> 


<Er> 
<td> 
<span id="DataList1l ct100 Label1">U 栓 </span> 
<span id="DataList1 ct100 Label2">140</span> &nbsp; 
<span id="DataList1 ct100 Label3">10</span> &gnbsp; 
<span id="DataList1_ ct100_ Label14">8G， 白 色 </span> 
</td> 
Ep 
<td> 
<span id="DataList1 ct101 Label1"> 内 存 </span> 
<span id="DataList1_ct101 Label2">160</span> &nbsp; 
<span id="DataList1 ct101 Label3">30</span> &nbsp; 
<span id="DataList1 ct101 Labe14">2G 内 存 </span> 
</td> 
/EE 
<table> 


从 以 上 代码 可 以 看 出 ，DataList 控件 生成 了 一 个 table，DataList 每 一 项 对 应 于 一 个 tr， 
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但 是 项 模板 里 每 个 控件 并 不 分 别 对 应 一 个 td, 而 是 所 有 控件 在 同一 个 td 中 ， 从 而 就 形成 了 
比较 混乱 的 布局 。 如 果 能 够 把 项 模板 中 每 一 个 Label 单独 放 到 一 个 td 中 ,那么 就 可 以 实现 
整齐 的 数据 表格 了 。 下 面 通 过 一 个 例子 演示 这 种 思路 。 

【 例 3-12】 以 规范 的 表格 显示 数据 。 

本 例 演 示 使 用 DataList 控件 以 整齐 的 带 表 头 的 表格 显示 商品 信息 列表 。 

(1) 创建 一 个 ASPNET 项 目 ， 按 照例 3-11 的 步骤 添加 数据 库 。 

(2) 添加 一 个 新 页 面 SimpleDataList2.aspx， 在 页 面 上 放置 一 个 DataList 控件 。 

(3) 在 代码 视图 中 编辑 DataList 控件 的 头 模板 ， 使 其 生成 商品 信息 表 头 。 代 码 如 下 ， 
注意 其 中 第 一 个 td 元 素 没 有 开始 标记 只 有 结束 标记 ， 而 最 后 一 个 td 元 素 没有 结束 标记 只 
有 开始 标记 。 这 是 因为 DataList 会 自动 生成 一 对 <tr><td></td></te> 标 记 ， 横 板 中 的 内 容 放 
在 这 一 对 标记 中 ， 通 过 把 模板 内 容 设置 为 以 下 代码 ， 模 板 代 码 与 自动 生成 的 代码 组 合 在 一 
起 ， 形 成 了 完整 的 5 个 td。 

<asp:DataList ID="DataList1" runat="server" RepeatLayout="Flow" > 

<HeaderTemplate> 

商品 编号 </td> 

<td> 商 品名 称 </td> ”<td> 商品 价格 </td> <td> 商 品 库存 </td> 

<td> 商 品 描述 

</HeaderTemplate> 

</asp:DataList> 

(4) 在 代码 视图 中 编辑 DataList 控件 的 项 模板 ， 项 中 每 一 个 Label 都 在 一 个 td 中 。 同 
样 的 道理 ， 第 一 个 td 元 素 没有 开始 标记 只 有 结束 标记 ， 而 最 后 一 个 td 元 素 没有 结束 标记 
只 有 开始 标记 。 

<ItemTemplate> 

<asp:Label ID="Label0" runat="server" Text="'<%®# Eval ("ProductID")%®>'> 

</asp:Label></td> 

<td> <asp:Label ID="Labell" runat="server" Text="'<$%# Eval("Product-— 
Name")%>'> </asp:Label> </td> 

<td><asp:Label ID="Label2" runat="server" Text="'<%# Eval ("Price")gs>" 
></asp:Label></td> 

<td><asp:Label ID="Label3" runat="server" Text="'<$%# Eval("Stock")%®> 
'></asp:Label></td> 

<td><asp:Label ID="Label4" runat="server" Text="'<%# Eval ("Description 
")%>'></asp:Label> 

</ItemTemplate> 

(5) 在 Page_Load 事件 中 编写 代码 以 绑 定数 据 ， 代 码 同 前 。 

(6) 运行 SimpleDataList2.aspx 页 面 ， 运 行 效果 
如 图 3.25 所 示 。 


DataList 用 伏 齐 的 表格 显示 数据 ~ Boxilla 了 ER 


中 入 区) 坦 看 历史 G) 书签 @)， 工具 利 W0D 
< rn sed 
Google | 4atalist 去 摘 br a EE 

ch > 
33.2 自 定义 布局 | petaList 用 整齐 的 表格 旦 示 数 舌 *| 上 
商品 编号 商品 名 称 | 商品 价格 | 商品 库存 商品 描述 
0001 | 唾 140 10 86， 白 色 


DataList 控件 可 以 自 定义 数据 显示 布局 , 从 而 设 ”一 重生 9 一 中 一 一 人 时 页 屏 


计 出 丰富 美观 的 页 面 。 在 购物 网 站 经 常 看 到 的 商品 。 es ew 一 人 
列表 页 面 ， 既 有 商品 的 各 种 图 片 ， 也 有 商品 的 相关 一。 PO 


文字 信息 ， 用 DataList 控件 能 够 很 方便 地 开发 出 这 畦 El 
种 图 文 并 茂 的 页 面 。 图 3.25 DataList 以 表格 形式 显示 数据 
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【 例 3-13】 显示 商品 列表 。 

本 例 用 DataList 控件 以 图 文 并茂 的 形式 显示 商品 列表 。 

(1) 创建 一 个 ASPNET 应 用 程序 , 把 前 面 所 创建 的 包含 Products 表 的 数据 库 复制 到 项 
目 中 App Data 目录 下 。 

(2) 修改 Products 表 结 构 ， 添 加 一 列 ImageUrl， 类 型 为 nvarchar(50)， 表 示 商 品 图 片 
路 径 。 在 项 目 中 添加 一 个 ProductImages 文件 夹 ， 用 于 存放 商品 图 片 。 

(3) 在 Products 里 添加 几 种 商品 ， 把 商品 图 片 复制 到 ProductImages 文件 夹 ， 并 且 设 
置 Products 表 中 ImageUrl 这 一 列 的 值 。 

(4) 在 项 目 中 添加 一 个 页 面 ProductListaspx， 在 页 面 上 放置 一 个 DataList 控件 。 

(5) 本 例 同时 显示 商品 图 片 和 文字 信息 ， 图 片 显示 在 左 侧 ， 文 字 在 右 侧 。 可 以 用 一 个 
table 来 实现 这 种 布局 ， 按 照 这 种 思路 设置 DataList 控件 的 项 模板 ， 代 码 如 下 : 

<asp:DataList ID="DataList1"” runat="server" HorizontalAlign="Center"> 

<ItemTemplate> 

<table> <tr> 

<td> <img src="ProductImages/<%#Eval ("ImageUrl1")%®>" 

alt="<%#Eval ("ProductName") gs>"” style="width:100px;" /> 

</td> 

<td><span style="font-weight:bold;"><%#Eval ("ProductName") %></span><br /> 

价格 : <%#Eval ("Price","{0:C}") 8s><br /> 

说 明 ; <%#Eval ("Description") %> </td> 

</tr> </table> 

</ItemTemplate> 

<ItemStyle BackColor="White" /> 


<AlternatingItemStyle BackColor="#ffffcc" /> 
</asp:DataList> 


(6) 在 Page_Load 事件 中 ， 编 写 代码 将 数据 绑 定 到 DataList。 

(7) 运行 ProductList.aspx 页 面 ， 运 行 效果 如 图 3.26 所 示 。 

(8) 由 于 商品 信息 太 少 ， 图 3.26 所 示 的 页 面 显得 很 细 长 ， 不 够 美观 ， 而 且 浪费 了 页 面 
很 多 空间 。 如 果 能 够 在 将 2 种 或 者 3 种 商品 信息 合并 在 一 行 显示 ， 那 么 效果 就 会 更 好 。 
DataList 控件 支持 这 个 功能 ， 通 过 设置 其 RepeatColumns 属性 可 以 指定 每 一 行 中 显示 的 项 
数目 。 此 处 将 RepeatColumns 设置 为 2， 代码 如 下 : 


<asp:DataList ID="DataListl" runat="server" HorizontalAlign="Center" 
RepeatColumns="2"> 


(9) 再 次 运行 ProductList.aspx 页 面 ， 运 行 效果 如 图 3.27 所 示 。 


六 件 中 旧 昌 四 查看 外 历史 G) 书签) 工具 CO 攻 赎 罗 HD WE EE ee 0 TR Hb 


< > 本 人 可 回 Frzzmnw 7 [En P oe ht 7 -HT 
ce J Cl 区 区 FTFT ER 


| wwzy1lecaaes-eaaetlist srs| | rrp | 


鸣 主板 
侣 价格 ， 140.00 价格 ， 字 580.00 
E :和 3 说 明 ，95， 白 色 , 黑色 ,红色 说 明 ， 华 硕 最 新 主板 
内 存 3 和 钥 ¥150.00 国生 于 ¥280.00 
价格 ， 站 160.00 3 .| 
| 和 说明，2c 内 存 ud 2 说明 ，25 内 存 请 和 和 ss 人 
滞 ， Cy | 
2 1, 258.00 a 
凑 ee Ty 二 denem 


习 加 
E33 Ea Ea ra 


图 3.26 商品 列表 页 面 图 3.27 每 行 2 种 商品 的 商品 列表 
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3.3.3 DataList 编辑 数据 


GridView 控件 实现 数据 编辑 和 删除 时 ， 只 需要 添加 一 个 CommandField， 再 配合 后 
台 代 码 即 可 。 而 DataList 控件 不 支持 CommandField， 所 以 DataList 实现 数据 删除 和 编辑 要 
比 GridView 稍微 复杂 一 些 。 

【 例 3-14】 DataList 编辑 和 删除 数据 。 

本 例 演 示 在 用 DataList 显示 、 编 辑 和 删除 商品 列表 。 

(1) 打开 例 3-13 所 创建 的 项 目 ， 或 者 新 建 一 个 ASPNET 项 目 ， 把 例 3-13 的 数据 库 和 
图 片 文件 复制 到 新 项 目 中 。 

(2) 在 项 目 中 添加 一 个 新 页 面 DataListEditaspx。 

(3) 为 DataList 创建 项 模板 以 显示 商品 列表 。 本 例 使 用 的 项 模板 与 例 3-13 不 同 ， 例 
3-13 每 种 商品 用 一 个 单独 的 table 显示 ， 而 本 例 所 有 商品 使 用 同一 个 table 显示 ， 每 种 商品 
是 table 中 的 一 个 tr。 为 了 实现 这 种 效果 ， 需 要 定制 DataList 控件 的 HeaderTemplate， 在 
ItemTemplate 中 也 要 注意 项 模板 中 td 标记 与 DataList 自动 生成 的 td 元 素 的 结合 。 另 外 ， 
DataList 中 还 包含 两 个 LinkButton 列 ， 分 别 用 于 编辑 和 删除 数据 。 

<asp:DataList ID="DataList1"” runat="server" HorizontalAlign="Center" 

oneditcommand="DataListl1 EditCommand" DataKeyField="ProductID" 
oncancelcommand="DataListl1 CancelCommand" 
ondeletecommand="DataList1 DeleteCommand" 
onupdatecommand="DataListl1 UpdateCommand"> 

<HeaderTemplate> 

</td><td></td><td></td><td> 

</HeaderTemplate> 

<ItemTemplate> 

<img src="ProductImages/<%#Eval ("ImageUrl1")®%>" alt="<%#Eval ("ProductName") 

%$>" style="width:100px;" /> </td> 

<td><span style="font-weight:bold;"> <%#Eval ("ProductName")%> </span><br 

?Es 

价格 : <s#Eval ("Price","{0:C}") %><br /> 说 明 : <%#Eval ("Description") $></td> 

<td><asp:LinkButton runat="server" id="editButton"” Text=" 编 辑 " Command- 

Name="edit" /></td> 

<td> 

<asp:LinkButton runat="server" id="deleteButton" Text=" 删 除 " CommandName 
="delete" 
OnclientClick="return confirm(' 确 实 要 删除 吗 ? ');" /> 


</ItemTemplate> 
</asp:DataList> 


在 上 述 代 码 中 ， 在 “删除 ”按钮 中 用 了 一 个 小 技巧 。“ 删 除 ” 按 钮 在 客户 端 会 调用 
JavaScript 语句 弹出 一 个 对 话 框 ， 请 用 户 确认 是 否 删除 。 如 果 用 户 选 择 取 消 删除 ， 则 不 会 产 
生 服 务 器 回 发 ， 也 不 会 删除 数据 。 

(4) 为 了 实现 DataList 编辑 数据 的 功能 ， 需 要 定制 EditItemTemplate 模板 。 在 模板 中 ， 
用 TextBox 控件 显示 商品 名 称 、 单 价 、 库 存 、 描 述 信息 ， 并 允许 用 户 编辑 ， 用 文件 上 传 控 
件 FileUpload 接受 用 户 上 传 的 产品 图 片 文件 。 在 编辑 项 模板 EditItemTemplate 中 ，“ 编 辑 ” 
按钮 变 成 了 两 个 按钮 : 一 个 “取消 ”和 一 个 “更 新 ”。 
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<EditItemTemplate> 

上 传 图 片 文件 : <br /> 

<asp:FileUpload ID="imageFile" runat="server" /><br /> 

(如 果 不 上 传 则 保持 原 有 图 片 不 变 ) 

</td> 
<td> 
名 称 : 
<asp:TextBox runat="server" ID="productName" Text="<%#Eval ("ProductName") 
ee 
价格 : <asp:TextBox runat="server" ID="productPrice" Text="'<%#Eval ("Price") 
EX 
说 明 : <asp:TextBox runat="server" ID="description" Text='<%#Eval ( 
“Description")®%>' /> <br /> 
库存 : <asp:TextBox runat="server" ID="stock" Text='<%#Eval ("Stock")%>' /> 
</td> <td> 
<asp:LinkButton runat="server" id="cancelButton" Text=" 取 消 " CommandName 
="cancel" /> 
<asp:LinkButton runat="server" id="updateButton" Text=" 保 存 " CommandName 
="update" /> 
</td> 
<td> <asp:LinkButton runat="server" id="deleteButton" Text=' 删 除 " 
CommandName="delete" 
OnClientClick="return confirm( "确实 要 删除 吗 ? ');" /> 
</EditItemTemplate> 


(5) 在 Page_Load 事件 中 编写 代码 绑 定数 据 。 
protected void Page Load(object sender, EventArgs e) 
{ 
if (!IsPostBack) 
{ 
bindList(); 
} 
} 
private void bindList() 
{ 
// 读 取 商 品 数据 并 绑 定 到 DataList 
SqlHelper db = new SqlHelper(); 
DbCommand command = db.GetSqlStringCommond ("select * from products"); 
DataTable table = db.ExecuteDataTable (command); 
DataList1.DataSource = table; 
DataListl1 .DataBind(); 
} 
(6) 在 DataList 的 EditCommand 事件 中 ， 将 事件 发 生 的 项 设置 为 编辑 状态 。 
protected void DataListl1 EditCommand (object source, DataListCommand 
EventArgs e) 
{ 
DataList1.EditItemIndex = e.Item.ItemIndex; // 设 置 当前 编辑 项 
bindList(); 


(7) 在 DataList 的 CancelCommand 事件 中 ， 取 消 所 有 项 的 编辑 状态 。 


protected void DataList1l CancelCommand (object source，DataListCommand 一 
EventArgs e) 


DataList1.EditItemIndex = -1; // 取 消 编辑 状态 
bindList(); 
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(8) 在 DataList 的 DeleteCommand 事件 中 ， 删 除 当 前 项 。 


protected void DataListl1 DeleteCommand (object source，DataListCommand- 
EventArgs e) 


{ 


| 


string key = DataList1.DataKeys[e.Item.ItemIndex] .ToString (); 
// 得 到 主键 
// 如 果 要 删除 的 项 当前 处 于 编辑 状态 ， 则 先 取消 编辑 状态 
if (DataList1.EditItemIndex == e.Item.ItemIndex) 
DataList1.EditItemIndex = -17 
string sql = "delete from products where productid=@id"; 
// 构 建 delete 语句 
SqlHelper db = new SqlHelper(); 
DbCommand command = db.GetSqlSstringCommond (sql); // 创 建 一 个 命令 
db.AddInParameter (command, "@id", DbType.String, key); 
// 向 命令 添加 参数 
db .ExecuteNonQuery (command); // 执 行 delete 命令 
bindList (); // 重 新 绑 定 数据 


(9) 在 DataList 的 UpdateCommand 事件 中 ， 将 用 户 输入 的 数据 保存 到 数据 库 。 要 注 
意 其 中 对 于 商品 图 片 文件 的 处 理 。 


protected void DataListl1 UpdateCommand (object source, DataListCommand-— 
EventArgs e) 


: 


// 获 取 用 户 输入 的 数据 
string name = (e.Item.FindControl ("productName") as TextBox) .Text; 
double price=double.Parse((e.Item.FindControl ("productPrice") as 
TextBox) .Text); 
double stock = double.Parse((e.Item.FindControl ("stock") as TextBox) 
.Text); 
string description = (e.Item.FindControl ("description") as TextBox). 
Text; 
string key = DataList1.DataKeys[e.Item.ItemIndex] .ToString (); 
// 获 得 上 传 文件 
FileUpload file = e.Item.FindControl ("imageFile") as FileUpload; 
string image = ""; 
// 如 果 用 户 传 了 文件 则 保存 到 ProductImage 目录 下 
if (file != null && file.HasFile) 
{ 
file.SaveAs (Server.MapPath ("~/ProductImages/" + file.FileName)); 
image = file.FileName; 
} 
// 构 建 update 语句 
string sql = "update Products set ProductName=@name, Price=@price," 
+ " Stock=@stock, Description=@description " 
+ (file.HasFile?", ImageUrl=@img ":" ") 
+ " where ProductID=@id"; 
/ /创建 一 个 命令 以 包含 刚才 创建 的 update 语句 
SqlHelper db = new SqlHelper(); 
DbCommand command = db.GetSqlSstringCommond (sql); 
// 向 update 命令 添加 各 个 参数 
db.AddInParameter (command, "@name", DbType.String, name); 
db.AddInParameter (command, "@price", DbType.Double, price); 
db.AddInParameter (command, "@stock", DbType.Double, stock); 
db.AddInParameter (command, "@description", DbType.String, description); 
db.AddInParameter (command, "@id", DbType.String, key); 
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if(file.HasFile) 
db.AddInParameter (command, "@img", DbType.Sstring,image); 
// 执 行 update 命令 ， 取 消 当前 项 的 编辑 状态 ， 重 新 绑 定数 据 
db.ExecuteNonQuery (command); 
DataListl1 .EditItemIndex = -1; 
bindList(); 
. 


(10) 运行 DataListEdit.aspx 页 面 ， 页 面 运行 界面 如 图 3.28 所 示 。 
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图 3.28 用 DataList 显示 和 编辑 商品 列表 


3.4 其 他 数据 绑 定 控件 


据 控 件 。 其 他 数据 绑 定 控 件 还 包括 Repeater、FormView、DetailsView 等 , 有 了 前 面 的 大王 
这 几 种 控件 的 使 用 也 很 容易 掌握 。 


3.4.1 Repeater 控件 


Repeater 控件 与 DataList 类 似 ， 也 是 以 数据 列表 的 形式 显示 数据 ， 也 需要 自 定义 项 模 
板 。 二 者 的 不 同 之 处 在 于 ， 在 最 终生 成 的 HTML 页 面 中 ，DataList 控件 会 自动 生成 少量 
局 代码 ， 如 table、tr、br， 而 Repeater 控件 本 身 不 会 生成 任何 HTML 代码 ， 所 以 Repeater 
通常 用 于 需要 严格 控制 生成 的 HTML 的 情况 下 。 下面 通过 一 个 例子 来 演示 Repeater 控件 的 
使 用 ， 这 个 例子 的 功能 也 是 显示 商品 列表 ， 通 过 将 这 个 例子 与 前 面 DataList 显示 商品 列表 
的 例子 比较 ， 可 以 更 明确 地 体会 二 者 的 异同 。 

【 例 3-15】 Repeater 控件 显示 商品 列表 。 

(1) 打开 例 3-13 所 创建 的 项 目 ， 或 者 新 建 一 个 ASPNET 项 目 ， 把 例 3-13 的 数据 库 和 
图 片 文件 复制 到 新 项 目 中 。 

(2) 添加 一 个 新 页 面 RepeaterSample.aspx， 在 页 面 上 放置 一 个 Repeater 控件 。 

(3) 本 例 要 用 一 个 table 来 显示 所 有 商品 信息 ， 每 个 商品 作为 一 个 tt。 在 Repeater 控件 
的 HeaderTemplate 中 ， 开 始 table 标记 。 


<HeaderTemplate> <table align="center"> </HeaderTemplate> 
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(4) 在 Repeater 控件 的 FooterTemplate 中 ， 结 束 table 标记 。 


<FooterTemplate> </table> </FooterTemplate> 


(5) 在 Repeater 控件 的 项 模板 IemTemplate 中 ， 生 成 一 个 tr 元素 显示 商品 信息 ， 其 中 


包括 两 个 td， 商 品 图 片 为 第 一 个 td， 其 他 文字 信息 为 第 二 个 ta。 要 注意 其 中 表示 是 否 有 货 
的 CheckBox 控件 的 数据 绑 定 表达 式 。 


<ItemTemplate> 

KU <td> 

<img src="ProductImages/<%#Eval ("ImageUTr1") %>" alt="<%#Eval ("Product-— 
Name") %>" style="width:100px;" /></td> 

<td><span style="font-weight:bold;"><%#Eval ("ProductName")®%></span><br /> 
价格 : <%#Eval ("Price","{0:C}") %><br /> 说 明 : <%#Eval ("Description") %><br /> 
是 否 有 货 : <asp:CheckBox runat="server" Enabled="false" 
Checked="<%#double.Parse (Eval ("Stock") .ToString())>0 %>' /> 

</td> </tr> 

</ItemTemplate> 


(6) 在 Page_ Load 事件 中 ， 编 写 代 码 绑 定数 据 。 


protected void Page Load (object sender, EventArgs e) 


{ 
if (!IsPostBack) 


{ 
// 查 询 所 有 商品 数据 并 绑 定 到 Repeater 控件 
SqlHelper db = new SqlHelper() 
DbCommand command = db.GetSqlStringCommond ("select * from products"); 


DataTable table = db.ExecuteDataTable (command); 
Repeater1.DataSource = table; 
Repeater1.DataBind() 


} 

(7) 运行 RepeaterSample.aspx 页 面 ， 运 行 效果 如 图 3.29 所 示 。 
文件 @) 编辑 于) 查看 W) 历史 G) 书签 @) 工具 0) 帮助 0 
/tt - le 


[> se EE EE EL E 
| netp:ymaecalhes--tersszle aspz| ~ | = 
是 有 代 ， 到 
内 存 | 
价格 ， 站 160. 00 
说 明 ，26 内 存 
和 请 是 否 有 货 ， 隐 
显示 器 
价格 ， 闻 1, 258. 00 
本 这 si 
是 否 有 货 ， 巾 了 


| 成 Ed 


图 3.29 ”Repeater 控件 显示 商品 列表 


3.4.2 ”DetailsView 控件 


E 如 其 名 称 所 指示 ，DetailsView 用 来 显示 数据 细节 。DetailsView 控件 用 表格 的 形式 


| 了 
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显示 数据 ， 每 次 只 能 显示 一 条 数据 ， 表 中 一 行 对 应 于 数据 源 的 一 个 字段 。DetailsView 支持 
分 页 ， 如 果 数 据 源 中 有 多 条 数据 ,那么 在 DetailsView 中 就 需要 使 用 多 页 显示 。DetailsView 
通常 用 来 显示 数据 详情 或 者 编辑 、 插 入 数据 。DetailsView 有 3 个 显示 模式 , 分 别 是 只 读 模 
式 、 编 辑 模式 和 插入 模式 ， 可 以 编程 对 这 3 种 模式 进行 切换 。 

【 例 3-16】 DeteailsView 显示 和 编辑 数据 。 

本 例 演示 使 用 DetailsView 显示 、 编 辑 、 删 除 、 添 加 商品 信息 。 

(1) 打开 例 3-13 所 创建 的 项 目 ， 或 者 新 建 一 个 ASPNET 项 目 ， 把 例 3-13 的 数据 库 和 
图 片 文件 复制 到 新 项 目 中 。 

(2) 添加 一 个 新 页 面 DetailsViewSample.aspx， 在 页 面 上 放置 一 个 DetailsView 控件 。 

(3) 将 DetailsView 控件 的 AutoGenerateDeleteButton 、AutoGenerateEditButton 、 
AutoGenerateInsertButton 属性 都 设置 为 tue， 使 DetailsView 自动 生成 添加 、 删 除 和 编辑 按 
钮 。 将 AllowPaging 属性 设置 为 tue 以 允许 分 页 。 此 时 DetailsView 代码 如 下 : 


<asp:DetailsView ID="detail" runat="server" DataKeyNames="ProductID" 
AllowPaging="true" 
AutoGenerateRows="false" AutoGenerateDeleteButton="true" 
AutoGenerateEditButton="true" AutoGenerateInsertButton="true" 
onpageindexchanging="detail PageIndexChanging" 
onitemdeleting="detail ItemDeleting" 
oniteminserting="detail ItemInserting" 
onitemupdating="detail ItemUpdating" 
onmodechanging="detail ModeChanging" > 

</asp:DetailsView> 


(4) 为 DetailsView 添加 各 个 字段 ， 设 置 各 字段 的 项 模板 和 项 编辑 模板 ， 其 中 商品 编 
号 ProductID 为 只 读 字段 , 商品 图 片 使 用 FileUpload 控件 编辑 。 完 成 这 些 设 置 后 DetailsView 
控件 代码 如 下 : 


<asp:DetailsView ID="detail" runat="server" DataKeyNames="ProductID" 
AllowPaging="true" 
AutoGenerateRows="false" AutoGenerateDeleteButton="true" 
AutoGenerateEditButton="true" AutoGenerateInsertButton="true" 
onpageindexchanging="detail PageIndexChanging" 
onitemdeleting="detail ItemDeleting" 
oniteminserting="detail ItemInserting" 
onitemupdating="detail ItemUpdating" 
onmodechanging="detail ModeChanging" > 


<%--Fields 标记 中 为 DetailsView 的 字段 列表 --%> 


<Fields> 
<s--asp:Bound 表示 数据 绑 定 列 ，DataField 为 要 绑 定 的 字段 名 ，Readon1ly 表示 只 读 
==%> 
<asp:BoundField DataField="ProductID" Readonly="true" HeaderText=" 商 品 
编号 ” /> 


<asp:BoundField DataField="ProductName" HeaderText=" 商 品名 称 "” /> 
<asp:BoundField DataField="Price"” HeaderText=" 价 格 " /> 
<asp:BoundField DataField="Stock" HeaderText=" 库 存 " /> 
<gs- -商品 图 片 是 一 个 模板 字段 ， 使 用 TemplateField 标记 --%> 
<asp:TemplateField HeaderText=" 商 品 图 片 "> 
<s--ItemTemplate 为 正常 显示 时 使 用 的 模板 --%> 
<ItemTemplate> 
<img src="ProductImages/<%#Eval ("ImageUTr1") $%>" alt="" style= 
swidth:200px2™ /> 
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</ItemTemplate> 
<%--EditItemTemplate 为 编辑 状态 时 使 用 的 模板 --%> 
<EditItemTemplate> 
<asp:FileUpload ID="imageFile" runat="server" /> 
</EditItemTemplate> 
</asp:TemplateField> 
</Fields> 
</asp:DetailsView> 


(5) 在 Page_Load 事件 中 编写 代码 绑 定数 据 。 


protected void Page _ Load (object sender, EventArgs e) 


if (!IsPostBack) 


{ 
} 


bindList(); 


} 
private void bindList() 
{ 
// 读 取 所 有 商品 数据 并 绑 定 到 DetailsVievw 控件 
SqlHelper db = new SqlHelper(); 
DbCommand command = db.GetSqlStringCommond ("select * from products"); 
DataTable table = db.ExecuteDataTable (command); 
detail.DataSource = table; 
detail.DataBind(); 
} 


(6) 在 DetailsView 控件 的 PageIndexChanging 事件 中 ， 编 写 代码 实现 翻 页 。 


protected void detail PageIndexChanging (object sender, DetailsViewPage- 
EventArgs e) 
| 
detail.PageIndex = e.NewPageIndex; // 切 换 到 新 页 
bingList (); // 重 新 绑 定数 据 


(7) 在 DetailsView 控件 的 ModeChanging 事件 中 实现 DetailsView 模式 切换 。 


protected void detail ModeChanging (object sender, DetailsViewModeEventArgs 
e) 
{ 
detail .ChangeMode (e.NewMode); 
bindList(); 
} 


(8) 在 DetailsView 控件 的 ItemDeleting 事件 中 ， 实 现 数据 删除 。 


protected void detail ItemDeleting(object sender, DetailsViewDelete-— 
EventArgs e) 
{ 


string key = detail.DataKey[0] .ToString(); // 得 到 主键 
string sql = "delete from products where productid=@id"; 
// 构 建 删除 SQL 语句 


SqlHelper db = new SqlHelper(); 

DbCommand command = db.GetSqlStringCommond (sql); 
db.AddInParameter (command, "@id", DbType.String, key); 
db.ExecuteNonQuery (command); 

detail .ChangeMode (DetailsViewMode .ReadOnly); // 切 换 到 只 读 模式 
bindList(); 
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(9) 在 DetailsView 控件 的 IemInserting 事件 中 ， 实 现 向 数据 库 中 添加 数据 。 


protected void detail ItemInserting (object sender，DetailsViewInsert- 
EventArgs e) 


1 


// 得 到 用 户 输入 的 数据 
string id = Convert .ToString(e.Values["ProductID"]) 
string name = Convert.ToString(e.Values["ProductName"]); 
double price = Convert.ToDouble(e.Values["Price"]); 
double stock = Convert.ToDouble(e.Values["Stock"]); 
string description = Convert.ToString(e.Values["Description"]); 
string image = ""; 
FileUpload file = detail.FindControl ("imageFile") as FileUpload; 
// 如 果 用 户 上 传 了 文件 ， 则 将 文件 保存 到 服务 器 
if (file.HasFile) 
| 
image = file.FileName; 
file.SaveAs (Server.MapPath ("~/ProductImages/" + file.FileName)); 
} 


// 构 建 SQL 语句 
string sql = "insert into Products " 
+ " (ProductID, ProductName, Price, Stock, Description, ImageUrl)" 


+ " values (@id, @name, Q@price, @stock, @description, @img)"; 
SqlHelper db = new SqlHelper(); 
DbCommand command = db.GetSqlStringCommond (sql); 
// 向 SqlCommand 对 象 添加 各 个 参数 
db.AddInParameter (command, "@name", DbType.String, name); 
db.AddInParameter (command, "@price", DbType.Double, price); 
db.AddInParameter (command, "@stock", DbType.Double, stock); 
db.AddInParameter (command, "@description", DbType.Sstring, description); 
db.AddInParameter (command, "@id", DbType.Sstring, id); 
db.AddInParameter (command, "@img", DbType.String, image); 


db.ExecuteNonQuery (command); // 执 行 insert 命令 
detail.ChangeMode (DetailsViewMode.ReadOnly) ; //detailsview 设置 为 只 读 模式 
bindList() 7 // 重 新 绑 定数 据 


(10) 在 DetailsView 的 ItemUpdating 事件 中 , 把 用 户 对 商品 信息 的 修改 更 新 到 数据 库 。 


protected void detail ItemUpdating(object sender, DetailsViewUpdate- 
EventArgs e) 


{ 
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// 读 取 用 户 输入 的 数据 
string id = detail.DataKey[0] .ToString() 
string name = Convert .ToString(e.NewValues ["ProductName"]) ; 
double price = Convert .ToDouble (e.NewValues ["Price"]) 
double stock = Convert .ToDouble (e.NewValues ["Stock"]) 
string description = Convert.ToString(e.NewValues["Description"]); 
string image = null; 
FileUpload file = detail.FindControl ("imageFile") as FileUpload; 
// 如 果 用 户 选择 了 上 传 文件 ， 则 将 文件 保存 到 服务 器 
if (file.HasFile) 
{ 
image = file.FileName; 
file.SaveAs (Server.MapPath ("~/ProductImages/" + file.FileName)); 
} 
/ /构建 update 语句 
string sql = "update Products set ProductName=@name, Price=@price," 
+ " Stock=@stock, Description=@description " 
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+ (file.HasFile ? ", ImageUr]l=@img "” : " ") 

+ " where ProductID=@id"; 
// 创 建 一 个 DbCommand 以 封装 update 语句 ， 并 添加 各 个 参数 
SqlHelper db = new SqlHelper(); 
DbCommand command = db.GetSqlSstringCommond (sql); 
db.AddInParameter (command, "@name", DbType.String, name); 
db.AddInParameter (command, "@price", DbType.Double, price); 
db.AddInParameter (command, "@stock", DbType.Double, stock); 


db.AddInParameter (command, "@description", DbType.String, description); 


db.AddInParameter (command, "@id", DbType.Sstring, id); 
if (file.HasFile) 
db.AddInParameter (command, "@img", DbType.String, image); 
// 执 行 update 命令 ， 重 新 绑 定数 据 
db .ExecuteNonQuery (command); 
detail.ChangeMode (DetailsViewMode.Readonly) ; 
bindList(); 


(11) 运行 DetailsViewSample.aspx 页 面 ， 运 行 效果 如 图 3.30 所 示 。 


Det nil:Vi er 显示 、 更 新 、 遇 除 、 击 加 政 反 ”0192 


3.4.3 FormView 控件 文件 四 名句 中 查看 历史) 书 答 因 工具 四 


鲜于 c/a - Mp 


Cm EE 


®-.@- 


FormView 控件 与 DetailsView 控件 比较 类 mews Ew am、 | 


加 


似 ， 也 是 用 于 显示 数据 源 中 的 单个 记录 。 商品 编号 Jo005 
FormView 控件 和 DetailsView 控件 之 间 的 差别 ee EE 
在 于 DetailsView 控件 使 用 表格 布局 ， 记 录 的 每 库存 |6 
个 字段 都 各 自 显示 为 一 行 。 而 FormView 控件 不 
指定 用 于 显示 记录 的 预定 义 布局 ,而 将 创建 一 个 商品 图 片 
模板 ， 其 中 包含 用 于 显示 记录 中 的 各 个 字段 的 
控件 。 
_ 、 Edit Delete New 
【 例 3-17】 FormView 显示 、 编 辑 、 删 除 |12s45678 
gi | 本 
数据 [天 El 
本 例 演 示 使 用 FormView 控件 显示 、 编 辑 和 


j 品 信 图 3.30 ”DetailsView 示例 
删除 商品 信息 。 ailsView 示 


(1) 打开 例 3-13 所 创建 的 项 目 ， 或 者 新 建 一 个 ASPNET 项 目 ， 把 例 3-13 的 数据 库 和 


图 片 文件 复制 到 新 项 目 中 。 


(2) 添加 一 个 新 页 面 FormViewSample.aspx， 在 页 面 上 放置 一 个 FormView 控件 。 


(3) 配置 FormView 的 项 模板 ， 使 用 表格 形式 显示 商品 图 片 和 信息 。 


<ItemTemplate> 

<table> 

<tr><td> 

<img src="ProductImages/<%#Eval ("ImageUT1") $%$>" alt="<%#Eval(" 
ProductName") %>" style="width:200px;" /></td> 


<td> <span style="font-weight:bold;"><%#Eval ("ProductName")®%></ 


span><br /> 
价格 : <s#Eval ("Price","{0:C}") %><br /> 说 明 : <%#Eval ("Description" 
/> 库存 : <%#Eval ("Stock") %></td> 
<td> <asp:LinkButton runat="server" id="editButton" Text=" 编 辑 " 
CommandName="edit" /> 


) $><br 
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</td> <td> 

<asp:LinkButton runat="server" id="deleteButton™" Text=" 删 除 " Command- 
Name="delete" 

OnClientClick="return confirm(' 确 实 要 删除 吗 ? ');" /> 

</td> </tr> 


</table> 
</ItemTemplate> 


(4) 配置 FormView 的 编辑 项 模板 ,用 TextBox 控件 编辑 商品 文字 信息 , 用 FileUpload 
控件 上 传 图 片 文件 。 


<EditItemTemplate> 


<table> <tr> <td> 
商品 图 片 <br /> 
<img src="ProductImages/<%#Eval ("ImageUT1") %>" alt="<%#Eval( 
"ProductName") $%>" style="width:200px;" /><br /> 
上 传 新 的 商品 图 片 : <br /> 

<asp:FileUpload ID="imageFile" runat="server" /><br /> 

(如 果 不 上 传 则 保持 原 有 图 片 不 变 ) </td> <tq> 
名 称 : <asp:TextBox runat="server" ID="productName" Text='<%#Eval 
("ProductName")%>"' /> <br /> 
价格 : <asp:TextBox runat="server" ID="productPrice" Text='<%#Eval 
("Price")®>" /> <br /> 
说 明 : <asp:TextBox runat="server" ID="description" Text="'<%#Eval 
("Description")%>' /> <br /> 
库存 : <asp:TextBox runat="server" ID="stock" Text="'<%#Eval ("Stock")%®>" 
/> </td> <td> 
<asp:LinkButton runat="server" id="cancelButton" Text=" 取 消 " 
CommandName="cancel" /> 
<asp:LinkButton runat="server" id="updateButton" Text=" 保 存 " Command- 
Name="update" /> 
</td> <td> 
<asp:LinkButton runat="server" id="deleteButton" Text=" 删 除 " 
CommandName="delete" 
OnClientClick="return confirm(' 确 实 要 删除 吗 ? ');" /> 
</td> </tr> 
</table> 


</EditItemTemplate> 


(5) 在 Page_ Load 事件 中 绑 定数 据 列 表 。 


protected void Page Load (object sender，EventRrgs e) 


{ 


} 


if (!IsPostBack) 
| 

bindList(); 
} 


private void bindList() 


{ 


1 


// 读 取 商 品 数据 并 绑 定 到 FormView 控件 

SqlHelper db = new SqlHelper(); 

DbCommand command = db.GetSqlStringCommond("select * from products"); 
DataTable table = db.ExecuteDataTable (command); 

formViewl .DataSource = table; 

formViewl .DataBind(); 


(6) 在 FormView 控件 的 PageIndexChanging 事件 中 ， 编 写 代码 实现 翻 页 。 
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protected void formView1 _ PageIndexChanging (object sender，EFormViewPage- 
EventArgs e) 
{ 
formView]l .PageIndex = e.NewPageIndex; 
bindList(); 
} 


(7) 在 FormView 控件 的 ModeChanging 事件 中 实现 FormView 模式 切换 。 


protected void formViewl ModeChanging (object sender, FormViewModeEventArgs 
e) 
{ 
formViewl .ChangeMode (e.NewMode) 
bingdList(); 
} 


(8) 在 FormView 控件 的 IemDeleting 事件 中 ， 实 现 数据 删除 。 


protected void formViewl ItemDeleting (object sender, FormViewDelete-— 
EventArgs e) 
{ 


string key = formView1.DataKey[0] .ToString(); // 得 到 主键 值 
string sql = "delete from products where productid=@id"; 
// 构 建 delete 语句 


SqlHelper db = new SqlHelper(); 
DbCommand command = db.GetSqlStringCommond (sql); 
db.AddInParameter (command, "@id", DbType.String, key); 


db.ExecuteNonQuery (command); // 执 行 删除 
formViewl .ChangeMode (FormViewMode .ReadOnly); 
bindList (); // 重 新 绑 定 


4 
(9) 在 FormView 控件 的 ItemUpdating 事件 中 ， 得 到 用 户 输入 的 信息 ， 并 保存 到 数 
据 库 。 


protected void formView1l ItemUpdating (object sender，EFormViewUpdate-- 
EventArgs e) 
{ 


// 获 取 用 户 输入 的 数据 
string id = formView1.DataKey[0] .ToString(); 
string name = (formView1.FindControl ("productName") as TextBox) .Text; 


double price = double.Parse ((formView1.FindControl ("productPrice") as 
TextBox) .Text); 
double stock = double.Parse( (formViewl.FindControl ("stock") as 
TextBox) .Text); 
string description = (formViewl.FindControl ("description") as TextBox). 
Text; 
string image = null; 
FileUpload file = formView]l.FindControl ("imageFile") as FileUpload; 
// 如 果 用 户 上 传 了 文件 ， 则 将 其 保存 到 服务 器 
if (file.HasFile) 
i 
image = file.FileName; 
file.SaveAs (Server.MapPath ("~/ProductImages/" + file.FileName)); 
} 
/ /构建 update 语句 
string sql = "update Products set ProductName=@name, Price=@price," 
+ " Stock=@stock, Description=@description " 
+ (file.HasFile ? ", ImageUr]l=@img " : " ") 


. 149 。 


第 1 篇 ASPNET 网 络 开发 关键 技术 


+ " where ProductID=@id"; 
// 创 建 一 个 命令 对 象 以 封装 update 语句 ， 并 对 该 命令 对 象 添 加 各 个 参数 
SqlHelper db = new SqlHelper(); 
DbCommand command = db.GetSqlStringCommond (sql); 
db.AddInParameter (command, "@name", DbType.Sstring, name); 


db.AddInParameter (command, "@price", DbType.Double, price); 
db.AddInParameter (command, "@stock", DbType.Double, stock); 


db.AddInParameter (command, "@description", DbType.String, description); 


db.AddInParameter (command, "@id", DbType.Sstring, id); 
if (file.HasFile) 


db.AddInParameter (command, "@img", DbType.String, image); 


// 执 行 update 语句 ， 然 后 重新 绑 定 数据 
db.ExecuteNonQuery (Command) 
formView1.ChangeMode (FormViewMode.Readonly) ; 
bindList(); 

， 


效果 如 图 3.31 所 示 。 


A 


(10) 运行 FormViewSample.aspx 页 面 ， 运 


Er TR ET 
We Oh/ 


四 
商品 图 片 


上 传 新 的 商品 副 片 ， 到 
(如 采 不 上 传 则 保持 原 有 医 片 不 变 ) 


图 3.31 FormView 示例 


3.$ 数据 源 控 件 


数据 源 控件 是 管理 连接 到 数据 源 及 读 取 和 写 入 数据 等 任务 的 ASPNET 控件 。 数 据 源 控 
件 不 呈现 任何 用 户 界面 ， 而 是 充当 特定 数据 源 〈 如 数据 库 、 业 务 对 象 或 XML 文件 ) 与 
ASPNET 网 页 上 的 其 他 控件 之 间 的 中 间 方 。 数 据 源 控 件 实现 了 丰富 的 数据 检索 和 修改 功 
能 , 其 中 包括 查询 、 排 序 、 分 页 、 筛 选 、 更 新 、 删 除 以 及 插入 。 数 据 源 控件 封装 了 ADO.NET 


的 功能 ， 如 连接 、 命 令 、 数 据 读 取 等 。 


3.5.1 SqlDataSource 控件 


SqlDataSource 控件 使 用 ADO.NET 类 与 ADO.NET 支持 的 任何 数据 库 进行 交互 。 这 类 
数据 库 包括 Microsoft SQL Server (使 用 System.Data.SqlClient 提供 程序 )、System.Data. 
OleDb、System.Data.Odbc 和 Oracle (使 用 System.Data.OracleClient 提供 程序 )。 使 用 


SqlDataSource 控件 ， 可 以 在 ASPNET 页 中 访问 和 操作 数据 ， 而 无 须 直接 使 


j ADO.NET 


类 。 只 需 提 供用 于 连接 到 数据 库 的 连接 字符 串 ， 并 定义 使 用 数据 的 SQL 语句 或 存储 过 程 即 


< 
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可 。 在 运行 时 ，SqlDataSource 控件 会 自动 打开 数据 库 连 接 ， 执 行 SQL 语句 或 存储 过 程 ， 


全 提示 : SqlDataSource 类 并 非 专门 用 于 连接 SQLServer 数据 库 ， 而 是 可 以 连接 各 种 
ADONET 支持 的 数据 源 。 有 人 曲解 了 SqlDataSource 数据 源 的 名 称 ， 认 为 SQL 
就 是 SQL Server， 其 实 这 里 的 SQL 是 泛 指 一 般 意 义 上 的 SQL， 而 非特 指 微软 公 
司 的 SQL Server 数据 库 。 


【 例 3-18】 SqlDataSource 控件 。 
本 例 使 用 SqlDataSource 作为 数据 源 ，GridView 作为 显示 控件 ， 显 示 商 品 信 息 列表 。 


(1) 创建 一 个 ASPNET 应 用 程序 ， 把 包含 Products 表 的 数据 库 复制 到 App_Data 目录 
下 。 然 后 在 项 目 中 添加 一 个 新 页 面 SqlDataSourceSample.aspx。 


(2) 在 页 面 上 放置 一 个 SqlDataSource 控件 。 单 击 控件 右上 角 智 能 按钮 ， 打 开 智 能 任 
务 面板 ， 选 择 “ 配 置 数 据 源 ” 如 图 3.32 所 示 。 
(3) 之 后 会 弹出 如 图 3.33 所 示 的 “配置 数据 源 ” 对 话 框 。 


本 过 皇 名 图 数 近东 
Dy 
T ia 
A 
| mo | su, | wr | ii | 
图 3.32 SqlDataSource 智能 任务 面板 图 3.33 “配置 数据 源 ” 对 话 框 


(4) 在 图 3.33 所 示 对 话 框 中 单 击 “ 新 建 连接 ”按钮 ， 则 弹出 如 图 3.34 所 示 的 “添加 
连接 ”对 话 框 。 

(5) 在 图 3.34 所 示 对 话 框 中 单 击 “ 更 改 ” 按 钮 ， 则 弹出 如 图 3.35 所 示 的 “更 改 数据 
源 ” 对 话 框 。 在 这 个 对 话 框 中 ， 列 出 了 SqlDataSource 所 支持 的 各 种 数据 源 ， 如 ODBC、 
SQL Server、Access 等 ， 可 以 选择 需要 连接 的 数据 源 类 型 。 


EE x 
谭 ， 或 单 击 “ 更 By" 选择 另 


数据 源 G): 
了 rosoft SGL Server 下 大 库 文 件 69 更改 c) | 
数据 库 文件 名 晰 建 或 现 有 名 称 ) 中 ) 

训 览 加 ). | 


登录 到 服务 器 
他 使 用 Windows 身份 验证 岂 ) 
个 合用 SQL Server 身份 验证 @@) 
用 Pew 


E22 


让 保 疗 宪 码 加 


wiv... | 
训 志 本 接 CD) 取消 网 


用 于 SQL Server 的 .NET Franew 加 
厂 始终 使 用 此 选择 &) 


图 3.34 “添加 连接 ”对 话 框 图 3.35 “更 改 数据 源 ” 对 话 框 


"Sl 
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(6) 本 例 需 要 连接 App_Data 目录 下 的 数据 库 文件 ， 所 以 从 图 3.35 所 示 对 话 框 中 选择 
“Microsoft SQL Server 数据 库 文件 ”， 单 击 “ 确 定 ” 按 钮 ， 然 后 从 弹出 的 对 话 框 中 选择 项 目 
中 的 数据 库 文件 。 接 下 来 ， 会 弹出 “配置 Select 语句 ”对 话 框 ， 如 图 3.36 所 示 。 在 这 个 对 
话 框 中 可 以 设置 查询 所 需要 的 SQL 语句 。 本 例 中 ， 选 择 Products 表 的 各 个 字段 。 

(7) SqlDataSource 控件 支持 自动 生成 编辑 数据 所 需 的 mnsert、Delete 和 Update 语句 ， 
从 而 允许 不 编写 代码 即 可 实现 数据 的 编辑 、 添 加 和 删除 。 在 图 3.36 中 , 单 击 “ 高 级 ”按钮 ， 
则 弹出 如 图 3.37 所 示 的 “高 级 SQL 生成 选项 ”对 话 框 , 在 其 中 选中 “生成 INSERT、UPDATE 
和 DELETE 语句 ” 单 选 按钮 。 


高 级 SQL 生成 选项 加 


可 以 生成 附加 的 INSERT、IFDATE 和 DELETE 语句 来 更 新 数据 源 。 


指定 宇和 ml 三 各 尿检 
指 这 开 和 目 过 训 加 ID) 
Er 
reat 习 
EL 
Er 


慷 入 或 TNSYRT、WIATE 和 WELETK 再 可 (G) 
A en wr 和 mr 再 名。 必须 选 定 所 有 主键 字段 才 


厂 合用 开放 式 并 发 (0) 
一 WDATE 和 DELETE 次 名 以 检测 上衣 记 录 加 吉 到 Dataset 中 以 来 数据 库 是 更 
际 i rt ead ei 和 


Cw |] ww | 
图 3.36 配置 Select 语句 图 3.37 高 级 SQL 生成 选项 


(8) 在 图 3.37 所 示 的 对 话 框 中 单 击 “ 确 定 ” 按 钮 以 后 ， 回 到 图 3.36 所 示 对 话 框 ， 一 
直 单 击 “ 下 一 步 ” 按 钮 ， 直 到 配置 结束 ， 单 击 “ 完 成 ”按钮 。 配 置 结束 以 后 ， 在 页 面 中 生 
成 以 下 代码 : 


<asp:SqlDataSource ID="SqlDataSourcel" runat="server" 
ConnectionSstring="<%$ ConnectionStrings:database %>" 
DeleteCommand="DELETE FROM [Products] WHERE [ProductID] = @ProductID" 
InsertCommand="INSERT INTO [Products] ([ProductID], [ProductName], 
[Price], [Stock], [Description]) VALUES (@ProductID, @ProductName, 
@Price, @Stock, @Description)" 
SelectCommand="SELECT [ProductID], [ProductName], [Price], [Stock], 
[Description] FROM [Products]" 
UpdateCommand="UPDATE [Products] SET [ProductName] = @ProductName, 
[Price] = @Price, [Stock] = @Stock, [Description] = @Description WHERE 
[ProductID] = @ProductID"> 
<DeleteParameters> 
<asp:Parameter Name="ProductID" Type="String" /> 
</DeleteParameters> 
<InsertParameters> 
<asp:Parameter Name="ProductID" Type="String" /> 
<asp:Parameter Name="ProductName" Type="String" /> 
<asp:Parameter "Price" Type="Double" /> 
<asp:Parameter "Stock" Type="Double" /> 
<asp:Parameter Name="Description" Type="String" /> 
</InsertParameters> 
<UpdateParameters> 
<asp:Parameter Name="ProductName" Type="String" /> 
<asp:Parameter Name="Price" Type="Double" /> 
<asp:Parameter Name="Stock" Type="Double" /> 
<asp:Parameter Name="Description" Type="String" /> 
<asp:Parameter Name="ProductID" Type="String" /> 
</UpdateParameters> 
</asp:SqlDataSource> 


cmaw |-sw>] 
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以 上 代码 主要 包括 3 部 分 内 容 ， 第 1 部 分 定义 了 连接 字符 串 ， 第 2 部 分 定义 了 Select、 
Jnsert、Update 和 Delete 语句 ， 第 3 部 分 定义 了 这 4 条 语句 用 到 的 参数 。 从 这 些 代 码 内 容 
来 看 ，SqlDataSource 封装 了 数据 库 连 接 和 增删 改 查 4 个 命令 ， 其 作用 相当 于 数据 适配器 
SqlDataAdapter。 

(9) 在 页 面 上 放置 一 个 GridView 控件 ,从 其 智能 任务 面板 中 选择 SqlDataSourcel 作为 
数据 源 ， 如 图 3.38 所 示 。 


图 3.38 为 GridView 选择 SqlDataSouce 作为 数据 源 


(10) 为 GridView 选中 SqlDataSourcel 作为 数据 源 以 后 ，GridView 会 根据 数据 源 中 的 
字段 自动 生成 列 ， 而 且 GridView 任务 面板 也 多 出 几 个 选项 ,包括 启用 分 页 、 启 用 编辑 、 启 
用 删除 等 ， 如 图 3.39 所 示 。 

(11) 为 了 演示 SqlDataSource 与 GridView 配合 工作 的 功能 ， 在 图 3.39 所 示 对 话 框 中 ， 
把 启用 分 页 、 启 用 排序 、 启 用 编辑 、 启 用 删除 和 启用 选 定 内 容 都 选中 。 为 GridView 各 个 字 
段 指定 中 文 列 标题 ， 浏 览 SqlDataSourceSample.aspx 页 面 ， 运 行 界面 如 图 3.40 所 示 。 从 运 
行 结果 可 以 看 到 ， 整 个 例子 中 没有 手工 编写 一 行 代码 ， 实 现 了 数据 的 显示 、 分 页 、 排 序 、 
删除 、 编 辑 功能 。 


文件 名。 韦 磺 久 亚 者 (0 夸克 避 ) 书签 一 工具 人 香 助 仙 ) 


[en 


加 | Griuvi ee tt 
EE 
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硬 委 1460 |)0 |5000 希 摊 PC 三 盘 
CPU 640 |12 |intel Core CFU 
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图 3.39 设 定 了 SqlDataSource 数据 源 以 后 的 GridView 图 3.40 ”SqlDataSource 示例 运行 界面 


3.5.2 数据 源 控件 参数 


在 很 多 情况 下 ， 页 面 上 并 不 需要 显示 所 有 数据 ， 而 需要 根据 用 户 的 选择 进行 数据 的 查 
询 和 和 显示。 例如， 在 网 上 书店 程序 中 ， 用 户 可 能 通过 输入 关键 字 〈 如 作者 、 书 名 等 ) 或 者 
选择 图 书 类 别 进行 查询 ， 程 序 需要 根据 用 户 输 入 的 条 件 执行 相应 的 检索 并 显示 数据 。 这 种 
情况 下 生成 的 SQL 查询 语句 以 用 户 输入 内 容 为 参数 。SqlDataSource 提供 了 灵活 强大 的 参 
数 化 查询 功能 ， 可 以 实现 这 种 需求 。 

【 例 3-19】SqlDataSource 参数 化 查询 。 


人 
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本 例 通过 SqlDataSource 参数 化 查询 实现 用 户 搜索 商品 的 功能 。 

(1) 创建 一 个 ASPNET 应 用 程序 ， 把 商品 信息 数据 库 和 商品 图 片 复制 到 项 目 中 。 

(2) 在 项 目 中 添加 一 个 页 面 ParameterizedQuery.aspx, 在 页 面 上 放置 一 个 SqlDataSource 
控件 和 一 个 GridView 控件 ， 代 码 如 下 : 


根据 名 称 搜索 : <asp:TextBox ID="productName" runat="server"></asp:TextBox> 

<asp:Button ID="Button1" runat="server" Text=" 搜 索 " /> 

<asp:GridView ID="GridViewl" runat="server"> 

</asp:GridView> 

<asp:SqlDataSource ID="SqlDataSourcel" runat="server"></asp:SqlDataSource> 

(3) 为 SqlDataSource 设置 数据 源 ， 大 体 步骤 同 例 3-18。 与 例 3-18 不 同 的 是 ， 当 出 现 
如 图 3.41 所 示 的 “配置 Select 语句 ”对 话 框 时 ， 选 中 Products 表 的 各 个 字段 后 ， 要 单 击 对 
话 框 右 侧 的 WHERE 按钮 。 


S| 7 ] 二 民 中 到 前 


图 3.41 配置 Select 语句 

(4) 之 后 会 弹出 如 图 3.42 所 示 的 “添加 WHERE 子 句 ”对 话 框 。 本 例 的 功能 是 需要 根 

据 用 户 输入 的 产品 名 称 进行 搜索 , 参照 图 3.42 在 这 个 对 话 框 中 进行 适当 配置 , 然后 单 击 “ 添 
加 ”按钮 ， 则 会 将 此 查询 条 件 添加 到 SqlDataSource 中 的 Select 语句 中 。 

ELITE JJ 


-id WERE 子 句 中 添加 一 个 或 多 个 条 件 。 可 以 为 等 个 条 件 指定 文本 值 或 参数 化 的 值 。 参 数 化 的 值 在 运行 时 根据 其 尿 性 获取 


列 四 : 参数 碌 性 
和 下 EST 
运算 罕 全 ); [pr oduct lne 局 
re 了 默认 值 D; 
源 @G) 
ER 到 | 
SQL 表达 式 : 值 - 
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图 3.42 添加 Where 子 句 
(5) SqlDataSource 配置 完成 后 ， 页 面 代码 变 成 以 下 内 容 。 


<asp:SqlDataSource ID="SqlDataSourcel" runat="server" 
ConnectionSstring="<%$ ConnectionStrings:database $>" 


. 154 。 
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SelectCommand="SELECT [ProductID]， [ProductName], [Price], [Stock], 
[Description] FROM [Products] WHERE ([ProductName] LIKE '%' + 
@ProductName + '$')"> 
<SelectParameters> 
<asp:ControlParameter ControlID="productName" Name="ProductName" 
PropertyName="Text" Type="String" /> 
</SelectParameters> 


</asp:SqlDataSource> 


(6) 将 GridView 控件 的 数据 源 设置 为 SqlDataSource 控件 , 并 且 为 各 个 字段 添加 标题 。 


<asp:GridView ID="GridViewl" runat="server" AutoGenerateColumns="False" 
DataKeyNames="ProductID" 


DataSourceID="SqlDataSourcel" PageSize="5" > 
<Columns> 
<asp:BoundField DataField="ProductID" HeaderText=" 商 品 编号 ”Read- 
Only="True" SortExpressio: ProductID" /> 
<asp:BoundField DataField="ProductName" HeaderText=" 商 品名 称 " 
SortExpression="ProductName" /> 
<asp:BoundField DataField="Price" HeaderText=" 价 格 " SortExpression= 


PFCew /> 
<asp:BoundField DataField="Stock" HeaderText=" 库 存 " SortExpression= 
"stock” /> 


<asp:BoundField DataField="Description" HeaderText=" 商 品 描述 " Sort- 
Expression="Description" /> 
</Columns> 


</asp:GridView> 

(7) 运行 页 面 ， 在 搜索 框 中 输入 关键 字 并 查询 ， 页 面 运行 效果 如 图 3.43 所 示 。 

(8) 本 例 到 此 为 止 实现 了 产品 搜索 功能 ， 下 一 个 步 又 在 GridView 中 添加 一 个 链接 ， 单 
击 此 链接 打开 一 个 新 页 面 ， 显 示 商 品 详情 ， 这 也 是 大 多 数 购物 网 站 都 具备 的 一 个 功能 。 在 
GridView 控件 中 添加 一 个 新 的 超 链接 列 ， 以 打开 商品 详情 页 面 。 在 GridView 控件 的 智能 
任务 面板 中 选择 “添加 字段 ” 则 打开 “添加 字段 ”对 话 框 ， 参 照 图 3.44 设置 各 个 选项 。 
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3.43 ”利用 SqlDataSource 参数 查询 商品 图 3.44 在 GridView 添加 超 链 接 字 段 


GridView 添加 超 链 接 字段 后 ， 生 成 如 下 代码 : 


i 
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<asp:HyperLinkField DataNavigateUrlFields="ProductID" 
DataNavigateUrlFormatString="ProductDetail.aspx?id={0}" HeaderText=" 


Text=" 查 看 " Target=" blank" /> 


上 述 代 码 含义 为 ， 当 在 浏览 器 中 单 击 此 链接 时 ， 会 在 新 窗口 中 打开 ProductDetail.aspx 


页 面 ， 并 将 当前 商品 ProductID 字段 的 值 作为 QueryString 参数 传递 给 新 页 面 。 


(9) 在 项 目 中 添加 一 个 新 的 页 面 ProductDetailaspx 以 显示 商品 详情 。 在 


ProductDetail.aspx 页 面 上 放置 一 个 SqlDataSource 以 从 数据 库 检 索 数 据 。 在 配置 SqlData 
Srouce 时 要 注意 添加 一 个 Where 条 件 ， 此 条 件 的 参数 来 源 于 QueryString, 具体 配置 界面 如 
图 3.45 所 示 。 


配置 结束 后 ，SqlDataSource 代码 如 下 : 


<asp:SqlDataSource ID="SqlDataSourcel" runat="server" 
ConnectionSstring="<%$ ConnectionStrings:database %>" 
SelectCommand="SELECT [ProductID], [ProductName], [Price], [Stock], 
[Description], [ImageUr1] FROM [Products] WHERE ([ProductID] = 
@ProductID) "> 
<SelectParameters> 

<asp:QueryStringParameter Name="ProductID" QueryStringField="id" 
Type="String" /> 

</SelectParameters> 

</asp:SqlDataSource> 


(10) 在 ProductDetail.aspx 页 面 上 放 一 个 FormView 以 显示 商品 详情 。 代 码 如 下 : 


<asp:FormView ID="FormView1" runat="server" DataSourceID="SqlDataSourcel" > 
<ItemTemplate> 
<table> 
a 
<td> <img src="ProductImages/<%#Eval ("ImageUT1") $%>" alt="<%#Eval 
("ProductName") %>" style="width:200px;" /></td> 
<td> 
<span style="font-weight:bold;"><%#Eval ("ProductName")®%></span><br /> 
价格 : <s#Eval ("Price","{0:C}") %><br /> 说 明 : <%#Eval ("Description") 
%><br /> 库存 : <%#Eval ("Stock") %></td> 
REFE> 
</table> 
</ItemTemplate> 
</asp:FormView> 


(11) 运行 ParameterizedQuery.aspx 页 面 , 并 单 击 一 个 商品 的 “查看 ”链接 , 打开 Product 


Detail.aspx 页 面 并 显示 该 商品 详细 信息 ， 如 图 3.46 所 示 。 
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图 3.45 为 SqlDataSource 添加 QueryString 参数 ”图 3.46 根据 QueryString 传 参 显 示 商 品 详情 页 面 
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3.5.3 ”其 他 数据 源 控 件 


除 SqlDataSource 数据 源 控件 以 外 ，ASPNET 还 主要 包括 以 下 数据 源 控件 。 

(1) EntityDataSource: 用 于 连接 到 实体 框架 Entity Framework 数据 源 。 

(2) AccessDataSource: 连接 到 Microsoft Access 数据 库 。 当 数据 作为 DataSet 对 象 返 
回 时 ， 支 持 排 序 、 筛 选 和 分 页 。 

(3) XmlDataSource: 连接 到 XML 文件 。 

(4) SiteMapDataSource: ASPNET 站 点 地 图 数据 源 控 件 。 

(5) LinqDataSource: 支持 语言 集成 查询 LINQ。 支 持 自动 生成 选择 、 更 新 、 插 入 和 删 
除 命令 。 该 控件 还 支持 排序 、 筛 选 和 分 页 。LINQ 将 在 本 书后 续 章 节 中 介绍 。 

(6) EntityDataSource: 支持 实体 框架 数据 源 。 支 持 自 动 生成 更 新 、 插 入 、 删 除 和 选择 
命令 。 该 控件 还 支持 排序 、 筛 选 和 分 页 。 实 体 框架 Entity Framework 将 在 本 书后 续 章 节 中 
介绍 。 

(7) ObjectDataSource: 允许 使 用 业务 对 象 或 其 他 类 ， 以 及 创建 依赖 中 间 层 对 象 管理 数 
据 的 Web 应 用 程序 。 支持 对 其 他 数据 源 控件 不 可 用 的 高 级 排序 和 分 页 方案 。 关于 多 层 结 构 
的 相关 知识 将 在 本 书后 续 内 容 中 介绍 。 

这 些 数据 源 控件 的 使 用 与 SqlDataSource 控件 大 同 小 异 ， 不 再 一 一 详细 介绍 。 


3.6 小 结 


本 章 介绍 了 ASPNET 数据 控件 。ASPNET 数据 控件 主要 包括 用 于 显示 数据 的 数据 绑 
定 控件 〈 如 GridView、DataList、DetailsView 等 ) 和 用 于 连接 数据 源 的 数据 源 控件 〈 如 
SqlDataSource、AccessDataSource、LinqDataSource 等 )。 这 些 数据 控件 封装 了 常用 的 数据 
显示 和 数据 访问 功能 , 提高 了 开发 效率 。 这些 控 件 在 各 种 ASPNET 数据 库 应 用 程序 中 广泛 
使 用 。 
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在 学 习 软 件 开发 技术 过 程 中 ， 很 多 人 都 会 遇 到 一 个 问题 : 技术 细节 学 了 一 堆 ， 但 是 却 
不 能 融会 贯通 ， 不 能 从 整体 上 把 握 这 些 具体 技术 细节 在 框架 中 的 地 位 及 互相 关系 ， 不 能 将 
各 种 技术 细节 综合 起 来 应 用 到 实际 项 目 中 。 为 了 避免 出 现 这 种 “只 见 树木 ， 不 见 森 林 ” 的 
情况 ， 本 章 将 综合 利用 前 面 几 章 学 习 过 的 ASPNET 编程 和 数据 访问 相关 技术 开发 一 个 网 
上 书店 ， 以 达到 技术 整合 和 项 目 锻炼 的 目的 。 


4.1 网 上 书店 整体 设计 


在 具体 编码 开发 网 上 书店 项 目 以 前 ， 先 要 分 析 此 项 目的 需求 ， 并 设计 出 网 站 的 整体 结 
构 。 为 了 叙述 方便 ， 将 本 章 所 开发 的 网 上 书店 称 为 “攀登 者 网 上 书店 ”。 本 节 将 分 析 这 个 
网 上 书店 的 主要 功能 需求 ， 并 设计 出 数据 库 结 构 。 


4.1.1 功能 需求 


攀登 者 网 上 书店 是 一 个 典型 的 网 上 书店 ， 功 能 与 其 他 书店 基本 相同 。 网 站 功能 整体 上 
可 以 分 为 两 大 部 分 : 前 台 功 能 和 后 台 功 能 。 网 站 前 台 应 该 包括 以 下 功能 。 

(1) 图 书信 息 展示 和 浏览 功能 。 网 站 能 够 显示 图 书 列表 ， 从 图 书 列表 可 以 导航 到 图 书 
详细 信息 。 

(2) 图 书 搜索 功能 。 用 户 可 以 根据 某 些 条 件 〈 如 书 名 、 作 者 、 出 版 社 等 ) 搜索 图 书 ， 
并 将 搜索 结果 按照 一 定 顺序 排序 〈 如 价格 、 销 售 量 等 ) 。 

(3) 购物 车 功能 。 用 户 在 浏览 图 书 过 程 中 ， 随 时 可 以 把 感 兴趣 的 图 书 放 到 购物 车 ， 并 
可 随时 查看 和 修改 购物 车 中 存放 的 图 书 。 

(4) 会 员 功能 。 用 户 可 以 在 网 上 注册 为 会 员 ， 可 以 填写 并 修改 会 员 相 关 信息 《如 联系 
电话 、 送 货 地 址 等 ) 。 

(5) 订单 功能 。 用 户 可 以 下 订单 购买 图 书 ， 并 在 一 定 条 件 下 〈 通 常 是 订单 尚未 处 理 时 ) 
可 以 修改 和 取消 订单 。 

(6) 评论 功能 。 用 户 可 以 对 图 书 进行 评论 。 

攀登 者 网 上 书店 后 台 主 要 包括 以 下 功能 。 

(1) 登录 和 用 户 权限 检测 。 只 有 管理 中 才 允 许 进入 后 台 管 理 页 面 ， 所 以 后 台 管 理 模块 
需要 检测 当前 用 户 是 否 拥有 足够 的 权限 。 

(2) 图 书 类 别管 理 。 浏 览 、 添 加 、 删 除 、 修 改 图 书 类 型 信息 。 

(3) 图 书 管理 。 浏 览 、 添 加 、 删 除 、 修 改 图 书信 息 ， 改 变 图 书 分 类 等 。 
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(4) 会 员 管理 。 可 以 浏览 、 添 加 、 删 除 、 修 改 会 员 信息 。 
单 管理 。 可 以 查询 订单 ， 改 变 订单 状态 (如 将 订单 状态 修改 为 已 发 货 ) ， 删 除 


(6) 评论 管理 。 可 以 查看 会 员 对 图 书 的 评论 ， 并 可 以 有 选择 的 删除 这 些 评论 。 


全 提示 : 限于 篇 幅 ， 本 章 只 给 出 了 较为 典型 和 重要 且 不 重复 的 代码 ， 主 要 包括 图 书 列表 显 
示 、 图 书 搜索 、 购 物 车 、 身 份 验 证 、 图 书信 息 维护 。 对 于 定单 和 评论 功能 ， 其 实 
现 思路 与 前 面 这 些 功能 类 似 ， 而 且 代 码 比 前 者 更 少 ， 本 章 没 有 给 出 具体 实现 。 


4.1.2 数据 库 结构 设计 


根据 前 面 对 攀 登 者 网 上 书店 功能 分 析 ， 可 以 设计 出 网 站 的 数据 库 结 构 。 网 站 主要 包括 

以 下 几 个 表 。 

(1) 类 别 表 Category。 

口 。、CategoryID: 类 别 ID，int 类 型 ， 主 键 ， 自 动 增长 列 。 

口 CategoryName: 类 别名 称 ，nvarchar(20) 类 型 。 

(2) 图 书 表 Book。 

口 BookID: 图 书 太 ，int 类 型 ， 主 键 ， 自 动 增长 列 。 

口 BookTitle: 图 书 标题 ，nvarchar(S0)。 

口 ”Authorl1、Author 2、Author3: 三 列 都 是 nvarchar(10) 类 型 ， 分 别 表示 第 一 作者 、 第 

二 作者 和 第 三 作者 。 

PublishID: 出 版 社 ID，int 类 型 ， 外 键 关 联 到 Pulisher 表 。 

PublishDate: 出 版 日 期 ，Datetime 类 型 。 

Price: 图 书 定价 ，float 类 型 。 

Discount: 折扣 比例 ，float 类 型 。 

Description: 图 书 描述 ，nvarchar(1000) 类 型 。 

Contents: 图 书目 录 ，text 类 型 。 

SaleSum: 该 书 总 销售 量 ，int 类 型 。 

ClickCount: 该 书 总 点 击 量 ，int 类 型 。 

SmallImage 和 BigImage: 图 书 的 大 小 图 片 。 

(3) 图 书 所 属 类 别 表 BookCategory。 

口 BookID: 图 书 呈 ，int 类型， 主键 ， 外 键 关联 到 Book 表 。 

口 CategoryID: 类 别 ID，int 类 型 ， 主 键 ， 外 键 关 联 到 Category 表 。 

全 提示 : 如 果 一 本 书 只 能 属于 一 个 类 别 ， 那 么 可 以 在 图 书 表 后 面 添 加 一 个 类 别 字段 以 标识 
图 书 所 属 的 类 别 。 但 是 ， 有 的 图 书 可 以 同时 属于 多 个 类 别 ， 这 种 情况 就 不 能 用 前 
面 的 方法 了 ， 而 是 需要 使 用 一 个 单独 的 表 来 描述 图 书 和 类 别 之 间 的 多 对 多 关系 。 

(4) 会 员 表 Member。 
口 MemberID: 会 员 ID，int 类 型 ， 主 键 ， 自 动 增长 列 。 


DDDGGODg90D 
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人 DOODODOCDD 


DODO 


口 


口 


思 回回: 回回 将 夯 所 癌 


MemberName: 会 员 名 称 ，nvarchar(20) 类 型 ， 唯 一 性 索引 约束 。 
Password: 会 员 登 录 密 码 ，nvarchar(20) 类 型 。 

Address: 送 货 地 址 ，nvarchar(100) 类 型 。 

Phone: 联系 电话 ，nvarchar(20) 类 型 。 

Email: 常用 电子 邮箱 ，nvarchar(20) 类 型 。 


5) 订单 表 BookOrder。 


OrderID: 订单 DD，int 类 型 ， 主键 ， 自 动 增长 列 。 

MemberID: 下 订单 的 会 员 ，int 类 型 ， 外 键 关联 到 Member 表 。 

OrderDate: 下 订单 日 期 ，Datetime 类 型 ， 默 认为 当前 时 间 。 

ProcessFlog: 订单 是 否 已 经 处 理 ，bit 类 型 ，0 表示 尚未 处 理 ，1 表示 已 经 处 理 〈 如 
配 货 发 货 等 ) 。 


(6) 订单 明细 表 OrderDetail。 


OrderID: 订单 编号 ，int 类 型 ， 外 键 关 联 到 Order 表 。OrderID 与 BookID 组 成 双 


BookID: 订单 所 订 图 书 ID，int 类 型 ， 外 键 关 联 到 Book 表 。OrderID 与 BookID 
组 成 双 主 键 。 


Price: 图 书 价格 ，float 类 型 。 
BuyNumber: 购买 数量 ，int 类 型 。 
TotalMoney: 总 金额 ， 等 于 PricexBuyNumber。 


7) 图 书评 论 表 BookRemark。 


BookID: 图 书 ID，int 类 型 ， 外 键 关联 到 Book 表 。 
MemberID: 会 员 ID，int 类 型 ， 外 键 关 联 到 Member 表 。 
Score: 图 书评 分 ，int 类 型 (1 一 5 之 间 ) 。 

Remark: 评论 内 容 ，nvarchar(500) 类 型 。 

RemarkDate: 评论 时 间 ，datetime 类 型 。 


4.1.3 网 站 整体 结构 


攀登 者 网 上 书店 是 一 个 简单 而 典型 的 购物 网 站 ， 拥 有 购物 网 站 的 主要 模块 ， 但 规模 不 
大 ， 功 能 不 复杂 ， 适 合作 为 练习 项 目 。 网 站 总 体 可 以 划分 为 两 大 组 成 部 分 : 网 站 前 台 和 网 
站 后 台 。 根 据 对 网 站 功能 需求 的 分 析 ， 人 攀登 者 网 上 书店 网 站 整体 结构 如 图 4.1 所 示 。 


4.2 网 上 图 书 前 台 功 能 实现 


攀登 者 网 上 书店 总 体 包括 前 台 和 后 台 两 大 模块 , 这 两 大 模块 之 间 互 相依 赖 , 联系 密切 。 
例如 ， 只 有 通过 后 台 增 加 了 图 书信 息 ， 才 能 够 在 前 台 页 面 搜索 和 显示 这 些 图 书信 息 ; 只 有 
在 前 台 下 了 订单 ， 后 台 才 能 够 对 订单 进行 处 理 。 本 节 将 介绍 前 台 功 能 的 实现 ， 在 4.3 节 再 
介绍 后 台 功 能 的 实现 。 


Is 
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一 图书 列表 会 员 注 册 
= 


A 
管理 录 窗 丰 
攀登 者 管理 员 登 录 修改 密码 


A 
类 别管 理 评论 管理 


[一 J 图书 管理 订单 管理 


会 员 管理 销售 统计 


图 4.1 网 上 书店 整体 结构 


4.2.1 母 版 页 和 主题 设计 


参照 目前 流行 的 网 上 书店 页 面 结 构 ， 攀 登 者 书店 页 面包 括 3 大 部 分 : 页 面 顶部 的 
LOGO、 广 告 和 顶部 导航 栏 、 页 面 中 间 主 要 内 容 、 页 面 底部 的 网 站 信息 。 其 中 页 面 顶部 内 
容 和 页 面 底 部 内 容 对 于 大 多 数 页 面 来 说 是 相同 的 ， 可 以 把 这 些 内 容 放 到 母 版 页 中 。 母 版 页 
BookShop.master 的 效果 如 图 4.2 所 示 。 


| .六 多 访 
Si 两 核 数 : | 最低! 送 T5GL 要 卡片 灯 E 9 校园 将 豆 | 所 大学) 


杜 计 首页 | 销售 排行 | 购物 车 | 宇 员 澄 录 | 我 的 订 疙 | 党 改 室 码 | 志 出 登录 | 。 挡 过 商品 * 搜索 | 训 按 过 


Ee 首页 | 关于 我 们 。 | 其 天 才 “| 。 玫 有 中 心 | 。 联系 我 们 “| 。 图书 目录 
是 mtxs rs 
S| pe 人 
上 竺 周二 1:30-19:30 属 六 至 由 a 
Ceprrisht 2010 芝 才 网 上 忆 计 . se 


图 42 网 上 书店 前 台 母 版 页 
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母 版 页 BookShop .master 代码 如 下 : 


<head runat="server"> 
<title> 攀 登 者 网 上 书店 </title> 
<link href="styles/bookshop.css" rel="stylesheet" type="text/css" /> 
<asp:ContentPlaceHolder ID="head" runat="server"> 
</asp:ContentPlaceHolder> 
<script type="text/javascript" language="javascript"> 


// 单 击 搜索 按钮 时 调用 此 函数 
function simpleSearch() { 
// 检 查 是 否 输入 了 检索 关键 字 
Var key = document .getElementById ("keyWord") .value; 
kh a | 
alert (' 请 输入 关键 字 再 搜索 ') ; 
return; 
} 
// 页 面 跳 转 到 简单 搜索 SimpleSearch .aspx， 并 通过 检索 字符 串 传递 参数 
window.location = "SimpleSearch.aspx?key=" + key; 
} 
</script> 
</head> 
<body> 
<form id="forml" runat="server"> 
<div id="page"> <%-- 最 外 层 div--%> 
<div id="header"> <g%-- 页 面 头 部 --%> 


<$-- 以 下 为 1ogo， 广 告 图 片 和 导航 链接 --g%> 
<img src="images/1ogo.jpeg"” alt=" 攀 登 者 网 上 书店 "” style="height:80px;" /> 
<img alt=" 大 学 优惠 " 

src="images/banner.jpg" style="height:80px;" /> 

<a href="ShopCartPage.aspx"> 

<img src="images/cart.gif" alt=" 查 看 购物 车 " /></a> <br /> 

<div style="padding:5px Opx; background-color:#e0f0ff; border:1lpx solid 
green;"> 

<a href="Default.aspx"> 本 站 首页 </a> | 

<a href="#"> 销 售 排行 </a> | 
<a href="ShopCartPage.aspx"> 购 物 车 </a> | 

<a 1 
<a 
<a 
<a href="#"> 退 出 登录 </a> 
<$-- 以 下 为 搜索 框 --g%> 
&nbsp; gnbsp; 搜索 商品 : <input type="text" id="keyWord" /> 
<a href="javascript:simpleSearch();"><img 
src="images/search.jpg" style="vertical-align:middle;"” alt=" 搜 索 "/></a> 
<a style="text-decoration:underline;" href="AdvanceSearch.aspx"> 高 级 搜索 


</a> 

</div> 
</div> <s--div header 结束 --%> 
<div style="clear:both;"> <%--- 页 面 主体 --%> 


<asp:ContentPlaceHolder ID="ContentPlaceHolder1"” runat="server"> 
</asp:ContentPlaceHolder> 

</div> 

<hr style="color:silver;" /> 

<s-- 以 下 为 页 脚 ， 包 含 链接 、 说 明文 字 、 图 片 、 网 站 备案 等 内 容 --s> 

<div id="footer"> 

<div style="float:left; 
<img src="images/logo.jpeg" alt="" /> 
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</div> 
<div style="margin-left: 10px7y float:left;" > 
<a href="Default.aspx"> 
<b> 首 页 </b></a>gnbsp; &nbsp; |&nbsp; gnbsp; 
<a href="#"><b> 关 于 我 们 </b></a> gnbsp; gnbsp; |&nbsp;&nbsp; 
<a href="#"><b> 诚 聘 英才 </b></a> 
<sup><img src="images/new2.gif"></sup>|é&nbsp; gnbsp; 
<a href="#"><b> 帮 助 中 心 </b></a> gnbsp; gnbsp;|&nbsp;é&nbsp; 
<a href="mailto:sun.j.1.studio@gmail.com"><b> 联系 我 们 </b></a>gnbsp; 
&nbsp; | Enbsp; gnbsp; 
<a href="Default.aspx"><b> 图 书目 录 </b></a> <br/> <br /> 
客户 服务 /电话 订购 : (010) 88888888<br/> 
<span class="gray12"><a target=" blank" href="#"> 在 线 客服 </a> 
<a target=" blank" href="#"> 联 系 客服 </a></span> 
<br/><span class="orangel3"> 工 作 时 间 周一 至 周 五 7:30-19:30 周 六 至 周 日 
9:00-17:30</span><br /> 
Copyright 2010 攀登 者 网 上 书店 .AL1 rights reserved </div> 
<div style="margin: 20px 30pt 20px auto float: right;"> 
<a target=" blank" href="http://www.315online.com. cn/member/315090031. 
html"> 
<img border="0" align="left" title=" 网 上 交易 保障 中 心 " style="margin: 10px 10px 
OpE OpEs" 
src="images/bz315.gif"/></a> 
<div style="margin-top: 1l0px; float:left; " > 
<a target=" blank" href="http://www.315o0nline.com.cn/member/315090031. 
html"> 网 上 交易 <br/> 保 障 中 心 </a> 
</div> 
<div style="float:left;"> 
<a target=" blank" href="http://www.hd315.gov.cn/beian/ view. 
asp?bianhao=010202001072000126"> 
<img border="0" title=" 备 案 " src="images/biaoshi.gif"/></a> </div> 
<div style="float:left; margin-top:10px;"> 
<a target=" blank" href="http://www.miibeian.gov.cn/"> 京 ICP 备 <br 
/>00011 号 </a> 
</div></div> 
</div> 
</div> 
</form> 
</body> 
</html> 


为 了 给 网 站 中 主要 控件 添加 统一 美观 的 外 观 效 果 ， 在 项 目 中 添加 一 个 主题 default， 
在 主题 中 添加 一 个 外 观 文件 ， 为 GridView 控件 、TextBox 控件 和 DropDownList 控件 定义 
外 观 ， 代 码 如 下 : 


<asp:GridView runat="server" CellPadding="4" ForeColor="#333333" 
GridLines="None" > 
<RowStyle CssClass="oddRow" /> 
<FooterStyle BackColor="#507CD1" Font-Bold="True" ForeColor="White" /> 
<PagerStyle BackColor="#2461BF" ForeColor="White" HorizontalAlign 
="Center" /> 
<SelectedRowStyle BackColor="#D1DDF1" Font-Bold="True" ForeColor 
三 "4333333% /> 
<HeaderStyle BackColor="#507CD1" Font-Bold="True" ForeColor="White" 
> 
<EditRowStyle BackColor="#2461BF" /> 
<AlternatingRowStyle CssClass="evenRow" /> 
</asp:GridView> 
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<asp:TextBox runat="server" CssClass="text"></asp:TextBox> 
<asp:DropDownList runat="server" CssClass="select"></asp:DropDownList> 


上 述 外 观 文件 代码 用 到 了 如 下 几 个 CSS 类 。 


.oddRow{ background-color:#EFF3FB;} // 表 格 奇数 行 样式 
.evenRow{background-color:White;} // 表 格 偶数 行 样式 
.hoverRow{background-color:#ffffdd;} // 表 格 鼠 标 划 过 时 样式 
-text{ border:solid 1lpx #9af;} // 文 本 框 样式 


-select{border:solid 1px;y background-color:#ffd; width:120px;}// 下 拉 框 样式 
在 web.config 文件 中 启用 default 主题 。 


<system.web> 
<pages styleSheetTheme="default"/> 
</system.web> 


4.2.2 网 站 中 的 通用 类 


在 整个 网 上 书店 的 开发 过 程 中 ， 会 用 到 一 些 共同 的 功能 ， 按 照 面 向 对 象 的 设计 思想 ， 
把 这 些 功能 封装 成 类 ， 不 但 能 够 实现 代码 复 用 ， 而 且 使 代码 结构 更 加 清晰 。 网 站 中 用 到 的 
类 主要 包括 图 书 类 Book、 图 书 数据 访问 类 BookDAL、 购 物 车 项 目 类 ShopCartItem、 购 物 
车 类 ShopCart 和 常量 容器 类 Constants。 


1. 图 书 类 Book 


Book 类 描述 了 图 书信 息 ， 其 中 包含 的 属性 与 数据 库 里 的 字段 基本 是 一 一 对 应 的 。 
public class Book 
{ 


public int id { get; set; } // 图 书 ID 
public string title { get; set; } // 图 书 标题 
public double price { get; set; } // 单 价 
public double discount { get; set; } // 会 员 折 扣 
public double realPrice { get; set; } // 打 折 后 价格 
public string authorl { get; set; } /作者 本 
public string author2 { get; set; } // 作 者 2 
public string author3 { get; set; } 7/ 作者 3 
public string smallImage { get; set; } // 小 封面 图 片 
public string bigImage { get; set; } // 大 封面 图 片 
public string content { get; set; } // 内 容 简 介 
public string description { get; set; } // 描 述 
public int clickCount { get; set; } // 点 击 数 量 
public int saleSum { get; set; } // 总 销售 量 
public int publisherID { get; set; } // 出 版 社 ID 
public string publisherName { get; set; } // 出 版 社 名 称 
public DateTime publishDate { get; set; } // 出 版 日 期 


} 
2. 图 书 数据 访问 类 BookDAL 
BookDAL 类 实现 了 对 于 图 书 相关 的 数据 库 操作 , 如 获取 某 一 特定 图 书信 息 、 按 照 条 件 
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搜索 图 书 等 功能 。 由 于 类 中 包含 的 都 是 方法 ， 而 没有 实体 属性 ， 所 以 把 此 类 声明 成 静态 类 
(static class) ， 类 中 所 有 方法 都 是 静态 方法 ， 从 而 允许 通过 类 名 直接 调用 方法 而 不 需要 实 
例 化 类 ， 方 便 了 方法 的 调用 。BookDAL 类 的 第 一 个 功能 是 根据 图 书 编号 获得 图 书信 息 。 


PE <summary> 

/// 根据 图 书 ID 从 数据 库 读 取 图 书信 息 

/// </summary> 

/// <param name="bookid"> 要 查询 的 图 书 ID</param> 
/// <returns> 从 数据 库 读 取 到 的 图 书信 息 </returns> 
public static Book getBookByID (int bookid) 


SqlHelper db = new SqlHelper (); 

// 构 建 查询 用 的 select 语句 

string sql = "select BookTitle,Author]l,Author2,Author3, " 
+" Price,Discount,RealPrice,PublisherID, PublisherName, PublishDate, " 
+" SmallImage,BigImage,Description,Contents,SaleSum,ClickCount " 
+" from ViewBookInfo where BookID=@id "; 

/ /创建 命令 并 添加 参数 

DbCommand command = db.GetSqlStringCommond (sql); 

db.AddInParameter (command, "@id", System.Data.DbType.Int32, bookid); 

/ /执行 命令 ， 得 到 数据 读 取 器 ， 读 取 数 据 

using (DbDataReader reader = db.ExecuteReader (command)) 

{ 
if (!reader.Read()) 

return null; // 如 果 没 有 数据 则 返回 空 

// 循 环 读 取 各 个 字段 ， 并 赋值 给 Book 类 的 实例 
Book book = new Book() 
book.id = bookid; 
book.title = reader["BookTitle"] .ToString(); 
book.authorl = Convert.ToString (reader["Author1"]); 
book.author2 = Convert.ToString (reader["Author2"]); 
book.author3 = Convert.ToString (reader["Author3"]); 
book.price = Convert.ToDouble (reader["Price"]); 
book.discount = Convert.ToDouble (reader["discount"]); 
book.realPrice = Convert.ToDouble (reader["RealPrice"]); 
book.publisherID = Convert.ToInt32 (reader["PublisherID"]); 
book.publisherName = Convert.ToString (reader["PublisherName"]); 
book.smallImage = Convert.ToString (reader["SmallImage"]); 
book.bigImage = Convert.ToString (reader["BigImage"]); 
book.saleSum = Convert .ToInt32 (reader["SaleSum"]); 
book.clickCount = Convert.ToInt32 (reader{["ClickCount"]); 
book.content = Convert.ToString(reader["Contents"]); 
book.description = Convert.ToString (reader["Description"]); 
book.publishDate = Convert.ToDateTime (reader ["PublishDate"]); 
reader.Close (); 
return book; 


上 

BookDAL 类 第 二 个 功能 是 根据 条 件 搜索 图 书信 息 。 图 书 的 查找 条 件 比较 复杂 , 可 以 根 
据 书 名 、 作 者 、 出 版 社 、 出 版 日 期 、 价 格 等 任意 组 合 条 件 查找 。 要 实现 这 个 功能 ， 可 以 定 
义 一 个 方法 ， 搜 索 条 件 作为 方法 的 参数 ， 代 码 如 下 : 


public static List<Book> search (string title, DateTime datel, DateTime date2, 
string author, 
string pulisher, double pricel, double price2); 


于 查询 条 件 太 多 ， 上 面 这 个 方法 参数 太 多 ， 方 法 签名 太 长 ， 不 方便 调用 。 为 了 改善 
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这 一 情况 , 可 以 用 一 个 类 SearchBookCriterion 来 封装 查询 条 件 , 则 search0) 方 法 只 需要 一 个 


参数 就 可 以 。 

// 搜索 图 书 条 件 

public class SearchBookCriterion 

| 
public string title=null; // 标 题 
public DateTime? datel=null, date2=null; // 出 版 日 期 
public string author=null; WE 
public string publisher=null; // 出 版 社 
public double? pricel=null, price2=null; // 价 格 区 间 


//public int? category=null; 


实现 图 书 搜索 时 ， 需 要 根据 查询 条 件 动态 构建 对 应 的 Select 语句 中 的 Where 条 件 。 实 
现 思路 为 ， 如 果 搜 索 条 件 中 题目 不 为 裤 ， 则 在 Where 子 句 中 添加 “BookTitle like @title”， 
如 果 搜 索 条 件 中 题目 为 室 ， 则 不 在 Where 子 句 中 添加 这 个 条 件 。 其 他 查询 条 件 也 需要 做 类 
似 处 理 。 这 个 功能 用 BookDAL 类 的 buildWhereString0 方 法 实现 ， 代 码 如 下 : 


/// <summary> 

/// 根据 查询 条 件 构建 SQL 语句 中 的 where 子 名 

/// </summary> 

/// <param name="criterion"> 查 询 条 件 </param> 

/// <returns> 构 建 好 的 where 子 句 </returns> 

private static string buildWhereString(SearchBookCriterion criterion) 


{ 
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string sgl = ™ 
string temp = null; 
// 如 果 标 题 不 为 室 ， 则 在 select 的 where 子 句 中 添加 标题 条 件 
if (!string.IsNullOrEmpty (criterion.title)) 
{ 
temp = string.Format(" and (BookTitle like '%{0}%') ", criterion. 
title); 
sql += temp; 
} 
// 如 果 作者 不 为 空 ， 则 在 select 的 where 子 句 中 添加 作者 条 件 
if (!string.IsNullOrEmpty (criterion.author)) 
{ 
temp = string.Format(" and (Authorl like '%{0}%' " 
.or Nuchor2 Tike SiO 本 
+ " or Author2 like '%{0}%' ) ", criterion.author); 
sql += temp; 
} 
// 如 果 日 期 范围 不 为 室 ， 则 在 select 的 where 子 句 中 添加 出 版 日 期 条 件 
if (criterion.datel.HasValue && criterion.date2.HasValue) 
{ 
temp = string.Format (" and (PublishDate between '{0}' and '{1}') ", 
criterion.datel .Value, criterion.date2.Value); 
sql += temp; 


} 
// 如 果 价 格 范围 不 为 室 ， 则 在 select 的 where 子 句 中 添加 价格 条 件 
if (criterion.pricel.HasValue && criterion.price2.HasValue) 
{ 
temp = string.Format(" and (Price between {0} and {1}) ", 
criterion.pricel, criterion.price2); 
sql += temp; 
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} 


// 如 果 出 版 社 不 为 室 ， 则 在 select 的 where 子 句 中 添加 出 版 社 条 件 

if (!string.IsNullOrEmpty (criterion.publisher)) 

! 
temp = string.Format(" and (PublisherName like '%{0}%') ", 
criterion.publisher); 


sql += temp; 


在 实现 图 书 搜索 时 ， 考 虑 到 符合 条 件 的 图 书 会 有 很 多 种 ， 通 常 需要 把 搜索 结果 分 页 返 
回 。 在 BookDAL 类 的 search() 方 法 中 ， 需 要 传递 相应 参数 pageIndex 和 pageSize 以 指定 返 
回 查 询 结果 的 第 几 页 和 每 页 记录 数 。 


MAA 
WA 
WA 
WA 
WA 
WA 
/// 
We 


<summary> 

根据 条 件 搜索 图 书 ， 并 分 页 返回 结果 

</summary> 

<param name="criterion"> 搜 索 条 件 </param> 


<param name="pageIndex"> 显 示 的 当前 页 </param> 
<param name="pageSize"> 页 面 大 小 </param> 
<param name="count"> 符 合 条 件 的 记录 总 数 </param> 
<returns> 指 定 页 的 数据 </returns> 


public static DataTable search (SearchBookCriterion criterion,int 
pageIndex, int pageSize,out int count) 


和 


全 提示 : 


string where=" "; 
if(criterion!=null) 

where = buildWhereString (criterion); // 得 到 查询 的 where 条件 
SqlHelper db = new SqlHelper(); 
// 用 select … + where … 构建 完整 的 查询 命令 
DbCommand command = db.GetSqlStringCommond ("select count (*) from 
ViewBookInfo where 1=1 "+where); 
// 得 到 满足 查询 条 件 的 图 书 总 数 
count = Convert .ToInt32 (db .ExecuteScalar (Command) ) 
// 分 页 查询 ， 查 询 指定 页 的 图 书 列表 
string sql = "SELECT BookID,BookTitle,Price,Discount, RealPrice, 
Authorl1, Author2, Author3," 

+ " SmallImage,PublisherName,PublishDate,™" 

+ "Row Number() over (order by BookID desc) as rownum " 

+ " FROM ViewBookInfo where (1=1) "; 
sql += where; 
string temp=null; 
sql = "WITH tempTable AS ("+ sql + ") "; 
temp = string.Format (" select * from tempTable where rownum between {0} 
nod 

(pageIndex - 1) * pageSize + 1, pageIndex * pageSize); 
sql += temp; 
command = db.GetSqlSstringCommond (sql); 
return db.ExecuteDataTable (command); 


上 述 代码 中 用 到 的 SQL Server 函数 Row_Number0 在 SQL Server 2005 以 上 版 本 
中 才能 够 支持 ， 如果 读者 所 用 的 数据 库 为 SQL Server 2000, 则 需要 使 用 其 他 分 页 
方法 ， 如 利用 Not In 和 SELECT TOP 分 页 ， 具 体 代码 可 在 网 上 查找 。 

上 述 代码 中 的 where (1=1) 是 动态 构建 select 语句 时 一 个 常用 小 技巧 。 如 果 
SELECT 语句 的 where 子 句 中 列 名 称 和 列 数量 都 不 固定 , 那么 在 构建 Where 条 件 
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时 ， 第 一 个 条 件 不 需要 添加 and， 后 面 的 条 件 都 需要 添加 and。 这 样 在 添加 查询 
条 件 时 ， 就 需要 不 断 判断 是 否 第 一 个 查询 条 件 ， 从 而 决定 是 否 需要 AND。 而 在 
SELECT 后 面 添加 WHERE (1=1) 很 巧妙 的 解决 了 这 个 问题 。 


购物 车 类 ShopCartltem 和 ShopCart 


ShopCartItem 描述 了 购物 车 中 的 一 种 图 书 ， 这 个 类 包括 图 书 相关 信息 和 所 购买 的 图 书 
数量 。 代 码 如 下 : 


public class ShopCartItem 


{ 


public ShopCartItem() { } 
public ShopCartItem(Book theBook, int num) 
{ 

book = theBook; 

number = num; 


} 


public Book book { get; set; } // 所 购买 的 图 书 
public int bookID 

{ get { return book.id; } } // 图 书 编号 
public string bookTitle / /图书 标 题 

{ get { return book.title; } } 

public double price // 价 格 

{ get { return book.price; } } 

public double discount // 折 扣 

{ get { return book.discount; } } 

public double realPrice // 实 际 购买 价格 
{ get { return book.realPrice; } } 

public string authorl // 作 者 1 

{ get { return book.authorl; } } 

public string author2 // 作 者 2 

{ get { return book.author2; } } 

public string author3 // 作 者 3 


{ get { return book.author3; } } 

public int number{get;set;} 

public double money // 金 额 = 单价 * 数 量 
{ 


get { return realPrice * number; } 


| 


ShopCart 类 表示 访问 网 站 的 一 个 用 户 的 购物 车 。ShopCart 类 是 一 个 ShopCartItem 的 集 
合 ,， 可 以 向 其 中 添加 图 书 ， 还 可 以 获取 统计 数据 ,如 购物 车 图 书 总 数 和 总 金额 等 。ShopCart 
类 代码 如 下 : 


// 购 物 车 ， 保 存在 Session 中 
public class ShopCart 
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// 购 物 车 中 现 有 的 图 书 列表 
private List<ShopCartItem> books=new List<ShopCartItem> () 7 
public List<ShopCartItem> booksInCart 
{ 
get 
return books; 


上 
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} 

/// <summary> 

/// 向 购物 车 添加 图 书 

/// </summary> 

/// <param name="bookid"> 要 添加 的 图 书 编号 </param> 
/// <param name="number"> 要 添加 的 图 书 数量 </param> 
public void addBook (int bookid, int number) 


{ 

// 查 找 要 添加 的 图 书 是 否 已 经 在 购物 车 中 存在 ， 如 果 存 在 则 只 修改 此 图 书 的 数量 
// 如 果 不 存在 则 需要 在 购物 车 中 添加 这 个 图 书 ， 并 设置 数量 
ShopCartItem item = books.Find(b => b.bookID == bookid); 
if (item != null) 
{ 

item.number += number; 

if (item.number == 0) 

books .Remove (item); 

return; 
Book book = BookDAL .getBookByID (bookid) ; // 从 数据 库 获取 图 书信 息 
if (book!=nul1) 

books.Add (new ShopCartItem(book, number)); 


} 
// 得 到 购物 车 中 所 有 图 书 总 金额 
public double totalMoney 
| 
get 
{ 
double money = 0; 
foreach (var item in books) 


money += item.money; 
} 
return money; 


} 


} 
// 得 到 购物 车 中 所 有 图 书 总 数量 
public int bookNumSum 
{ 
get 
{ 
int sum = 0; 
foreach (var item in books) 
{ 
sum += item.number; 
} 
return sum; 


} 
} 
// 得 到 购物 车 中 图 书 种 类 数 


public int bookCount 
{ 

get { return books.Count; } 
} 


4. 常量 容器 类 Constants 


在 网 上 书店 项 目的 多 个 页 面 中 都 需要 访问 一 些 常量 ， 例 如 ， 需 要 访问 Session 的 一 些 
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键 值 ， 以 取得 Session 中 存储 的 相应 对 象 。 把 这 些 常量 封装 到 一 个 类 中 集中 存储 ， 既 方便 
了 代码 的 编写 和 维护 ， 又 能 够 避免 编程 时 Session 键 值 拼写 出 错 。 


internal static class Constants 


和 


internal const string SessionRdmin = "AdminLogin"; 
internal const string SessionMember = "MebmerLogin" 7 
internal const string SessionShopCart = "ShoppingCart"; 


4.2.3 网 书 列表 用 户 控件 


在 网 上 书店 的 多 个 页 面 中 都 需要 显示 图 书 列表 ， 为 了 实现 复 用 ， 可 以 把 图 书 列 表 做 成 
个 用 户 控件 BookListControl， 在 其 中 定义 好 图 书 显 示 布 局 ， 实 现 相 应 的 事件 处 理 程序 。 
当 在 页 面 中 需要 显示 图 书 列表 时 ， 只 需要 使 用 此 控件 即 可 。 

BookListControl 用 户 控件 使 用 Repeater 控件 显示 图 书 列表 。 了 Repeater 控件 的 
ItemTemplate 包含 左右 两 大 部 分 ， 左 侧 为 图 书 封面 图 片 ， 右 侧 为 图 书 详细 文字 描述 ， 还 有 
放 到 购物 车 的 按钮 购买 数量 。 图 书 封面 为 超 链接 ， 用 户 可 以 通过 单 击 图 书 封面 转 到 图 书 详 
情 页 面 。BookListControl 控件 显示 效果 如 图 4.3 所 示 。 


设计 模式 可 复 用 面向 对 象 软件 的 基础 
作者 ，Erich GammaRichard HelmJohn| 出 版 社 ， 电 子 工业 出 版 社 | 出 版 日 期 ，1998-1-1 0:00:00 
定价 。 衬 35.00 | 会 员 价 ， 茸 28. 00| 折扣 ，0.8 


购买 数量 ，1 四 加 入 风光 车 查看 详情 


C# 本 质 论 
作者 ， Mark Jichaelis| 出 版 社 ， 清华 大 学 出 版 社 | 出 版 日 期 ，2009-11-1 0:00:00 
定价 。 等 85.00 | 会 员 价 ， 壮 63.75| 折扣 :0.75 


we 


图 43 图 书 列表 用 户 控 件 BookListControl 


BookListControl 控件 代码 如 下 : 


<asp:Repeater ID="Repeaterl" runat="server" OnItemCommand="Repeaterl 
ItemCommand"> 
<HeaderTemplate> 
<table> 
</HeaderTemplate> 
<ItemTemplate> 
< 
<td> 
<a href="BookDetail.aspx?id=<%#Eval ("BookID")®%>"> 
<img src="BookImages/<%#Eval ("SmallImage") $>" alt="<%#Eval ("BookTitle") 
Sy 
style="width:100px;" /> </a> 
</td> 
<td> 
<h3> <%#Eval ("BookTitle") %> </h3> 
作者 : <%#Convert -ToString (Eval ("Author1")) + Convert.ToString (Eval 


~ 
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("Author2")}) 十 
Convert .ToString (Eval ("Author3"))%>| 
出 版 社 : <s#Eval ("PublisherName") %$>| 出 版 日 期 : <s#Eval ("PublishDate") %><br 
1><br’ /> 
定价 : <span style="color:Blue;"> <%#Eval ("Price", "{0:C}") %$></span> | 
会 员 价 : <span style="color:red;"><%#Eval ("RealPrice","{0:C}")%></span>| 
折扣 : <s#Eval ("Discount")%><br /><br /> 
购买 数量 : <asp:TextBox ID="buyNum" runat="server" Text="1"></asp:TextBox> 
<asp:ImageButton runat="server" ID="buyButton" ImageUrl="~/images/buy. 
png" Height="25" 
CommandName="buy" CommandArgument="<%#Eval ("BookID") $>' ImageAlign="top" /> 
<span style="margin-left:50px;"> 
<a href="BookDetail.aspx?id=<%#Eval ("BookID")%>"> 查 看 详情 </a> 
</span> 
</td> 
</tr> 
</ItemTemplate> 
<FooterTemplate> 
</table> 
</FooterTemplate> 
</asp:Repeater> 


图 书 列表 控件 BookListControl 将 在 多 个 页 面 中 使 用 ， 需 要 显示 各 种 图 书 数据 ， 所 以 
BookListControl 控件 本 身 不 能 确定 其 数据 源 ， 而 是 需要 定义 一 个 接口 ， 允 许 外 部 页 面 为 其 
绑 定数 据 源 。 在 BookListControl 控件 中 添加 一 个 bindData 方法 实现 这 个 功能 。 


public void bindData (DataTable table) 
| 
Repeater1l.DataSource = table; 
Repeater1l.DataBind() 
} 


当 用 户 在 图 书 列表 中 单 击 “ 加 入 购物 车 ”按钮 时 , 会 触发 Repeater 控件 的 TemCommand 
事件 ， 在 此 事件 中 编写 代码 ， 获 取 用 户 输入 的 数量 ， 将 用 户 所 选 的 图 书 添 加 到 Session 中 
的 购物 车 。 代 码 如 下 。 


protected void Repeaterl ItemCommand (object source, RepeaterCommand— 
EventArgs e) 
{ 
if (e.CommandName == "buy") 
TextBox text=e.Item.FindControl ("buyNum") as TextBox; 
// 找 到 数量 TextBox 控件 
int n; 
if (!int.TryParse (text.Text, out n)) 
Dh 
int id = int.Parse(e.CommandArgument.ToString()); 
// 从 Session 中 获取 购物 车 ， 并 将 图 书 添加 到 其 中 
if (Session[Constants.SessionShopCart] == null) 
Session[Constants.SessionShopCart] = new ShopCart (); 
ShopCart cart = Session[Constants.SessionShopCart] as ShopCart; 
cart.addBook (id, n); 
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4.2.4 网 站 首页 


当 用 户 登 录 网 上 书店 首页 时 ， 应 该 向 用 户 展示 某 些 图 书信 息 ， 例 如 最 新 推出 的 图 书 或 


者 最 近 销 售 量 最 大 的 图 书 。 同 时 ， 首 页 上 应 该 允许 用 户 方便 地 找到 自己 所 需要 的 图 书 ， 应 


该 向 
页 中 


一 个 


如 下 


j 户 提供 按照 关键 字 搜索 图 书 的 功能 。 其 中 ， 搜 索 图 书 按钮 已 经 包含 在 母 版 页 中 ， 首 
需要 做 的 主要 工作 是 显示 推荐 的 图 书 列表 ， 本 例 将 显示 最 新 添加 的 图 书 。 
(1) 在 数据 库 中 添加 视图 ViewBookInfo 以 获取 图 书 详细 信息 。 


CREATE VIEW ViewBookInfo 
AS 
SELECT dbo.Book.BookID, dbo.Book.BookTitle, dbo.Book.Authorl, dbo.Book. 
Author2, 
dbo.Book.Author3, dbo.Book.PublisherID, dbo.Book.Price, dbo.Book. 
Discount, 
dbo.Book.PublishDate, dbo.Book.Description, dbo.Book.Contents, 
dbo.Book.SaleSum, 
dbo.Book.ClickCount, dbo.Book.SmallImage, dbo.Book.BigImage, 
dbo.Book.RealPrice, 
dbo.Publisher .PublisherName 
FROM dbo.Book INNER JOIN 
dbo.Publisher ON dbo.Book.PublisherID = dbo.Publisher.PublisherID 


(2) 在 数据 库 中 添加 一 个 存储 过 程 以 分 页 读 取 最 新 添加 的 图 书 。 


CREATE PROCEDURE dbo.GetNewBook 


@pageIndex int=1, /* 要 返回 的 页 码 */ 
@pageSize int=10, /* 页 面 大 小 */ 
@bookCount int output /* 输 出 参数 , 记录 总 数 */ 


RS 
declare @min int , @max int; 
set @min=(@pageIndex-1)*@pageSize+l; 
set @max=@pageIndex*@pageSize; 
select bookCount=count (*) from Book; 
WITH NewBooks AS 
(SELECT 
BookID, BookTitle, Price,Discount,RealPrice,Author],Author2, Author3, 
SmallImage, PublisherName, PublishDate, 
Row Number() over (order by BookID desc) as rownum 
FROM ViewBookInfo) 
select 
BookID, BookTitle, Price,Discount,RealPrice,Author]l,Author2, Author3, 
SmallImage, PublisherName, PublishDate 
from NewBooks where rownum between @min and @max ; 
RETURN 


(3) 使 用 BookShop.master 作为 母 版 页 添加 一 个 新 页 面 Default.aspx， 并 在 页 面 上 放置 
图 书 列表 用 户 控 件 BookListControl 和 一 个 第 三 方 分 页 控件 AspNetPager， 页 面 代 码 


<%@ Register Assembly="AspNetPager" Namespace="Wuqi.Webdiyer" TagPrefix= 
"webdiyer" %> 

<#sQ@ Register src="BookListControl.ascx" tagname="BookListControl" 
tagprefix="ucl" 和 > 

<asp:Content ID="Content1" ContentPlaceHolderID="head" runat="server"> 
</asp:Content> 

<asp:Content ID="Content2" ContentPlaceHolderID="ContentPlaceHolderl" 
runat="server"> 
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<ucl:BookListControl ID="bookList" runat="server" /> 
<webdiyer:AspNetPager ID="AspNetPager]l" runat="server"> 
</webdiyer:AspNetPager> 

</asp:Content> 


(4) 在 Default.aspx 页 面 的 Page Load 事件 中 编写 代码 显示 第 一 页 最 新 图 书信 息 。 
protected void Page Load (object sender, EventArgs e) 


| 
if (!Page.IsPostBack) 


// 页 面 首次 加 载 时 ， 设 置 分 页 控件 默认 大 小 为 10， 当 前 页 码 为 1， 并 绑 定 第 1 页 数据 
AspNetPager]l .PageSize = 10; 
AspNetPager]l .CurrentPageIndex = 1; 
bindData (); 
} 
private void bindData() 
1 
SqlHelper db = new SqlHelper(); 
// 调 用 GetNewBook 存储 过 程 
DbCommand command= db.GetStoredProcCommond ("GetNewBook"); 
// 为 存储 过 程 添加 参数 : 2 个 输入 参数 〈 页 号 和 页 面 大 小 ) ，1 个 输出 参数 (记录 总 数 ) 
db.AddInParameter (command, "@pageIndex", DbType.Int32, AspNetPagerl. 
CurrentPageIndex); 
db.AddInParameter (command, "@pageSize", DbType.Int32, AspNetPager!l. 
PageSize); 
db.AddOutParameter (command, "@bookCount", DbType.Int32, 4); 
DataTable table= db.ExecuteDataTable (command); 
bookList.bindData (table); 
// 设 置 AspNetPager 分 页 控件 的 记录 总 数 
AspPNetPager1l1.RecordCount = Convert.ToInt32 (Command. Parameters 
["@bookCount"] .Value) ; 


(5) 在 AspNetPager 控件 的 PageChanged 事件 中 重新 绑 定 新 页 面 的 数据 。 


protected void RspNetPagerl PageChanged (object sender, EventArgs e) 


{ 
bindData (); 


} 
(6) 运行 Default.aspx 页 面 ， 运 行 效果 如 图 4.4 所 示 。 
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图 4.4 攀登 者 网 上 书店 首页 
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4.2.5 购物 车 


攀登 者 网 上 书店 可 以 随时 从 页 面 顶部 链接 进入 购物 车 页 面 ， 查 看 和 修改 购物 车 中 的 图 
书 商品 。 由 于 一 个 用 户 对 应 于 一 个 购物 车 ， 而 且 需 要 随时 访问 购物 车 ， 因 此 把 购物 车 存放 
在 Session 中 是 一 个 很 好 的 解决 方案 。 当 进入 购物 车 页 面 时 ， 需 要 从 Session 而 不 是 数据 库 


中 读 取 购物 车 数据 。 
(1) 使 用 BookShop.master 作为 母 版 页 添加 一 个 新 页 面 ShopCartPage.aspx。 


(2) 在 ShopCartPage 页 面 上 放置 一 个 GridView 以 显示 购物 车 数据 。GridView 的 最 后 
两 列 分 别 是 模板 列 和 超 链 接 列 ， 模 板 列 中 用 一 个 TextBox 控件 显示 了 图 书 数量 ， 用 户 可 以 


编辑 此 数量 。 用 户 如 果 单 击 超 链 接 列 会 转移 到 图 书 详情 页 面 。 


<asp:GridView ID="GridViewl" runat="server" DataKeyNames="bookID" 
AutoGenerateColumns="false" ShowFooter="true" 
onrowupdating="GridViewl RowUpdating" onrowcreated="GridViewl] 
RowCreated" > 

<s-- 设 置 GridView 的 空 模板 ， 当 没有 数据 时 就 显示 此 模板 的 内 容 --%> 

<EmptyDataTemplate> 
<div class="dashedborder" style="color:navy; padding:20px 30px; 
font-size:20px;"> 
你 的 购物 车 中 没有 商品 。 
</div> 
</EmptyDataTemplate> 

<g%-- 以 下 为 GridView 的 各 个 列 --%> 

<Columns> 

<asp:BoundField DataField="bookID" HeaderText=" 图 书 编号 "/> 


<asp:BoundField DataField="bookTitle" HeaderText=" 图 书 题目 " ItemStyle- 
Width="260"/> 


<asp:BoundField DataField="price" DataFormatSstring="{0:C}" HeaderText=" 


定价 " Itemstyle-Width="80"/> 

<asp:BoundField DataField="discount" HeaderText=" 会 员 折扣 " ItemStyle- 
Width="80"/> 

<asp:BoundField DataField="realPrice" DataFormatString="{0:C}" 
HeaderText=" 会 员 价 格 " ItemStyle-Width="80"/> 


<asp:BoundField DataField="number" HeaderText=" 购 买 数 量 " ItemStyle- 
Width="60"/> 


<asp:BoundField DataField="money" DataFormatSstring="{0:C}" HeaderText=" 


金额" 
ItemStyle-Width="80"/> 


<%—— <asp:TemplateField HeaderText=" 作 者 " ItemStyle-Width="200"> 
<ItemTemplate> 


<%#Convert .ToString (Eval ("authorl")) + Convert.ToString (Eval ("author2")) 


Convert .ToString (Eval ("author3"))%> 

</ItemTemplate> 

</asp:TemplateField> --%> 

<asp:TemplateField HeaderText=" 修 改 数量 " > 

<ItemTemplate> 

<asp:TextBox runat="server" ID="buyNum" Text="'<%#Bind("number")%®>" 

Width="80" /> 

<asp:Button 

ID="Button1l" runat="server"” Text=" 修 改 " CommandName="update" 
CommandArgument="<%#Bind ("bookID")%®>" /> 
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</ItemTemplate> 

</asp:TemplateField> 

<xasp:HyperLinkField Text=" 查 看 ”HeaderText=" 详 情 " 
DataNavigateUrlFields="BookID" 
DataNavigateUrlFormatString="BookDetail.aspx?id={0}" /> 

</Columns> 

</asp:GridView> 


(3) 在 Page_ Load 事件 中 ， 获 取 Session 中 保存 的 购物 数据 ， 并 显示 在 GridView 控 
件 中 。 


protected void Page Load (object sender, EventArgs e) 
{ 
if(!IsPostBack) 
bindData (); 
. 
private void bindData() 


{ 
// 从 session 中 获取 购物 车 ， 并 将 其 中 的 图 书信 息 显示 在 Gridview 中 
if (Session[Constants.SessionShopCart] == null) 
return; 
ShopCart cart = Session[Constants.SessionShopCart] as ShopCart; 
GridViewl .DataSource = (Session[Constants.SessionShopCart] as 
ShopCart) .booksInCart; 
GridViewl .DataBind(); 
} 


(4) 在 GridView 的 RowCreated 事件 中 ， 生 成 购物 车 汇总 数据 并 显示 在 GridView 的 
Footer 中 。 


protected void GridView1l RowCreated (object sender, GridViewRowEventArgs e) 
{ 
// 汇 总 数据 在 页 脚 中 ， 如 果 当 前 行 不 是 页 脚 ， 则 不 进行 处 理 ， 方 法 返回 
if (e.Row.RowType != DataControlRowType.Footer) 
return; 
// 清 除 页 脚 原 有 单元 格 ， 创 建 一 个 跨越 所 有 列 的 大 单元 格 
int n=e.Row.Cells.Count; 
e.Row.Cells.Clear (); 
TableCell cell = new TableCell (); 
cell.ColumnSpan = n; 
ShopCart cart = Session[Constants.SessionShopCart] as ShopCart; 
// 在 单元 格 中 显示 汇总 信息 
cell.Text = string.Format ("购物 车 中 总 计 {0} 种 {1} 本 图 书 ， 总 金额 {2:C}"， 
cart .bookCount, cart .bookNumSum, cart .totalMoney); 
e.Row.Cells.Add (cell); 
} 


(5) 当 用 户 修改 购物 车 数量 时 ， 会 触发 GridView 的 RowUpdating 事件 ， 在 此 事件 中 
修改 Session 中 保存 的 购物 车 图 书 数量 。 


protected void GridViewl RowUpdating(object sender, GridViewUpdate-— 
EventArgs e) 
{ 


string id = e.Keys[0].ToSstring(); 

// 得 到 购买 数量 TextBox 控件 

TextBox control = GridViewl .Rows[e.RowIndex] .FindControl ("buyNum") as 
TextBox; 

ShopCart cart = Session[Constants.SessionShopCart] as ShopCart; 

int num=int.Parse (Control .Text); 


= 


第 1 篇 ASPNET 网 络 开发 关键 技术 


Var book=cart.booksInCart-Find(b => b.bookID == int.Parse(id)); 


// 查 找 购物 车 中 此 图 书 
if (num > 0) 
book.number = num; 
else 
cart .booksInCart.Remove (book); 
bindData(); // 重 新 绑 定 数据 


} 
(6) 购物 车 页 面 运行 界面 如 图 4.5 所 示 。 
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4.2.6 简单 搜索 


攀登 者 网 上 书店 可 以 方便 地 根据 书 名 或 者 作者 搜索 图 书 。 用 户 在 页 面 顶部 的 搜索 框 中 


输入 关键 字 ， 单 击 “ 搜 索 ” 按 钮 ， 即 可 查找 书 名 或 者 作者 名 字 
索 框 和 搜索 按钮 是 在 母 版 页 BookShop.master 中 定义 的 ， 


中 包含 此 关键 字 的 图 书 。 
当 单 击 “ 搜 索 ” 按 钮 时 ， 会 调用 


搜 


母 版 页 中 以 下 JavaScript 代码， 检查 用 户 是 否 输入 了 关键 字 ， 并 将 搜索 关键 字 作 为 
QueryString 参数 传递 给 SimpleSearch.aspx 页 面 , 在 SimpleSearch.aspx 页 面 中 实现 图 书 搜索 


<script type="text/javascript" language="javascript"> 
function simpleSearch() { 


// 获 取 用 户 输入 的 关键 字 ， 验 证 是 否 为 室 ， 如 果 为 空 则 提示 
Var key = document getElementById ("keyWord") .value; 
if (!key) { 


alert (' 请 输入 关键 字 再 搜索 ') ; 


return; 


} 
// 跳 转 到 简单 搜索 页 面 ， 并 将 搜索 关键 字 作 为 参数 传递 
window.location = "SimpleSearch.aspx?key=" + key; 
} 
</script> 


(1) 在 数据 库 中 添加 一 个 存储 过 程 SimpleSearch， 实 现 简单 搜索 功能 。 
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CREATE PROCEDURE dbo.SimpleSearchBook 
/* 根据 书 名 或 者 作者 搜索 图 书 */ 


@title author nvarchar (50), /* 书 名 或 作者 */ 
@pageIndex int=1, /* 页 码 */ 
@pageSize int=10, /* 页 面 大 小 */ 
@bookCount int output /* 查 找到 的 总 数量 */ 


// 计 算 指定 页 面 的 最 小 行 号 和 最 大 行 号 

declare @min int , @max int; 

set Q@min=(@pageIndex-1)*@pageSizet+l; 

set Q@max=@pageIndex*@pageSize; 

// 执 行 查 询 

set @title author="'%'+@title author+'s%®" 

select ebookCount=count (*) from Book 

where (BookTitle like @title author) 

or (authorl like @title author) 

or (author2 like @title author) 

or (author3 like @title author); 

WITH NewBooks AS 

(SELECT BookID,BookTitle, Price,Discount, RealPrice,Authorl, Author2, 
Author3, 

SmallImage, PublisherName, PublishDate, 

Row Number() over (order by BookID desc) as rownum 

FROM ViewBookInfo where BookTitle like '%'+@title author+'%' ) 
select BookID,BookTitle, Price,Discount,RealPrice, Authorl, Author2, 
Author3, 

SmallImage, PublisherName, PublishDate 

from NewBooks where rownum between @min and @max; 


(2) 使 用 BookShop.master 作为 母 版 页 添加 一 个 新 页 面 SimpleSearch.aspx， 并 在 页 面 
上 放置 一 个 图 书 列表 用 户 控 件 BookListControl 和 一 个 第 三 方 分 页 控件 AspNetPager， 页 面 
代码 如 下 : 


<%@ Register Assembly="AspNetPager" Namespace="Wuqi.Webdiyer" TagPrefix= 
"webdiyer" %> 
<%@ Register src="BookListControl.ascx" tagname="BookListControl" 
tagprefix="ucl" %> 
<asp:Content ID="Contentl" ContentPlaceHolderID="head" runat="server"> 
</asp:Content> 
<asp:Content ID="Content2" ContentPlaceHolderID="ContentPlaceHolderl" 
runat="server"> 
<ucl:BookListControl ID="bookList" runat="server" /> 
<webdiyer:AspNetPager ID="AspNetPagerl" runat="server" 
onpagechanged="AspNetPagerl PageChanged"> 
</webdiyer:AspNetPager> 
</asp:Content> 


(3) 在 Page_Load 事件 中 , 获取 QuerySting 中 的 查询 关键 字 , 调用 存储 过 程 搜索 图 书 ， 
并 显示 结果 。 


protected void Page Load(object sender, EventArgs e) 
{ 
if (!IsPostBack) 
{ 
string key = Request.QueryString["key"]; // 得 到 查询 关键 字 
if (string.IsNullOrEmpty (key)) return; 
AspNetPagerl .CurrentPageIndex = 1; 
bindData(); // 执 行 查询 
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1 
private void bindData() 
1 
string key = Request.QueryString["key"]; // 得 到 查询 关键 字 
SqlHelper db = new SqlHelper(); 
// 调 用 SimpleSearchBook 存储 过 程 查询 图 书 
Var command=db.GetStoredProcCommond ("SimpleSearchBook"); 
db.AddInParameter (command, "@title author", System.Data.DbType. 
String, key); 
db.AddInParameter (command, "@pageIndex", System.Data.DbType.Int32, 
AspNetPagerl] .CurrentPageIndex); 
db.AddInParameter (command, "@pageSize", System.Data.DbType.Int32, 
AspNetPagerl] .PageSize); 
db.AddOutParameter (command, "@bookCount", System.Data.DbType.Int32,4); 
Var table=db.ExecuteDataTable (command); 
// 将 查询 到 的 结果 绑 定 到 GridView， 并 设置 分 页 控件 的 总 记录 数 
bookList.bindData (table) 
int count= Convert .ToInt32 (command.Parameters["@bookCount"] .Value) ; 
AspNetPager1.RecordCount = count; 
// 若 查询 到 的 总 记录 数 为 0， 则 提示 没有 符合 条 件 的 数据 
if (AspNetPagerl .RecordCount == 0) 
{ 
ClientScript.RegisterStartupScript (this.GetType(), "nobooks", 
"<script> 没 有 符合 条 件 的 图 书 。</script>"); 


, 
(4) 在 AspNetPager 控件 的 PageChanged 事件 中 显示 新 页 面 数据 。 


protected void AspNetPagerl PageChanged(object sender, EventArgs e) 
bindData (); 
由 


4.2.7 ”高 级 搜索 


在 高 级 搜索 页 面 中 ， 用 户 可 以 根据 书 名 、 作 者 、 出 版 社 、 出 版 日 期 、 价 格 区 间 ， 以 及 
这 些 条 件 的 任意 组 合 搜索 图 书 。 

(1) 使 用 BookShop.master 作为 母 版 页 添加 一 个 新 页 面 AdvanceSearch.aspx。 

(2) 在 AdvanceSearch.aspx 页 面 上 放置 用 于 输入 各 个 搜索 条 件 的 TextBox 控件 。 


<div class="dashedborder"> 
标题 : <asp:TextBox ID="bookTitle" runat="server"/> 
&nbsp; 作 者 : <asp:TextBox ID="author" runat="server"/> 
&nbsp; 出 版 社 : <asp:TextBox ID="publisher" runat="server"/> <br /> 
价格 范围 : <asp:TextBox ID="pricel" runat="server"></asp:TextBox> 至 <asp: 
TextBox ID="price2" runat="server"></asp:TextBox> 
&nbsp7 
出 版 日 期 : <asp :TextBox ID="datel" runat="server"></asp:TextBox> 至 <asp: 
TextBox ID="date2" runat="server"></asp:TextBox> 
&nbsp; <asp:Button ID="Buttonl" runat="server"” Text=" 搜 索 " Width="80" 
onclick="Buttonl Click" /> 
</div> 
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(3) 在 页 面 上 放置 一 个 图 书 列表 用 户 控件 BookListControl 和 一 个 第 三 方 分 页 控件 
AspNetPager。 


<%@ Register Assembly="AspNetPager" Namespace="Wuqi.Webdiyer" TagPrefix= 
"webdiyer" %> 
<%@ Register src="BookListControl.ascx" tagname="BookListControl" 
tagprefix="ucl" $%> 
<ucl:BookListControl ID="bookList" runat="server" /> 
<webdiyer:AspNetPager ID="AspNetPagerl" runat="server" 
onpagechanged="AspNetPagerl1 PageChanged"> 
</webdiyer:AspNetPager> 


(4) 在 “搜索 ”按钮 的 Click 事件 中 ， 根 据 用 户 输 入 的 搜索 条 件 查 找 并 显示 符合 条 伯 
的 第 一 页 数据 。 要 注意 把 查询 条 件 保存 到 Session 中 ， 从 而 可 以 在 用 户 翻 页 时 重新 获取 并 
继续 使 用 此 查询 条 件 。 


private void bindData() 
{ 


上 


tt 


// 从 Session 中 获取 查询 条 件 
SearchBookCriterion criterion = Session["SearchCriterion"] as 
SearchBookCriterion; 
int index = AspNetPagerl]l .CurrentPageIndex; 
int size = AspNetPager] .PageSize; 
int count; 
// 调 用 数据 访问 层 代码 执行 搜索 操作 ， 并 将 结果 绑 定 到 控件 
DataTable table = BookDRAL .search (criterion，index，size，out count); 
BookListControl1.bindqData (table); 
AspNetPager1.RecordCount = count; 
} 
protected void Button1l Click(object sender, EventArgs e) 
{ 
// 根 据 用 户 输入 的 查询 条 件 构建 一 个 SearchBookCriterion 实例 
SearchBookCriterion criterion = new SearchBookCriterion () 
if (!string.IsNullOrEmpty (bookTitle.Text)) 
criterion.title = bookTitle.Text; 
if (!(string.IsNullOrEmpty (pricel.Text) || string.IsNullOr- 
Empty (price2.Text))) 


criterion.pricel = double.Parse(pricel.Text); 
criterion.price2 = double.Parse (price2.Text); 
} 
if (!string.IsNullOrEmpty (author.Text)) 
criterion.author = author.Text; 
if (!string.IsNullOrEmpty (publisher.Text)) 
criterion.publisher = publisher.Text; 
if (!(string.IsNullOrEmpty (datel.Text) || string.IsNullOrEmpty 
(date2 .Text) ) ) 


criterion.datel DateTime.Parse (datel .Text); 
criterion.date2 = DateTime.Parse (date2.Text) .AddDays 
AddSeconds (-1); 

} 

Session["SearchCriterion"] = criterion; 

// 将 当前 页 码 设置 为 1， 根 据 新 条 件 重 新 绑 定 数据 

AspNetPagerl .CurrentPageIndex = 1; 

bindData (); 

| 


(5) 在 AspNetPager 的 PageChanged 事件 中 显示 新 的 一 页 数据 。 


= 


第 1 篇 ASPNET 网 络 开发 关键 技术 


protected void AspNetPagerl PageChanged (object sender, EventArgs e) 
{ 
bindData(); 


(6) 高 级 搜索 页 面 AdvanceSearch.aspx 运行 界面 ， 如 图 4.6 所 示 。 
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4.3 网 上 书店 后 台 功 能 实现 


管理 员 登 录 网 上 书店 后 台 ， 可 以 对 图 书信 息 进 行 全 面 管理 ， 如 添加 、 删 除 图 书 ， 修 改 
图 书 文字 资料 ， 上 传 图 书 封面 图 片 等 。 为 了 维护 网 站 数据 安全 ， 只 能 经 过 登录 验证 的 管理 
员 才 能 访问 后 台 管理 页 面 ， 在 实现 后 台 页 面 时 需要 考虑 身份 验证 的 问题 。 为 了 便于 进行 身 
份 验证 ， 所 有 后 台 管 理 页 面 都 在 网 站 根 目录 下 的 admin 文件 夹 中 。 


4.3.1 用 户 身份 验证 模块 


实现 用 户 身份 验证 最 简单 的 方法 就 是 用 户 登录 后 把 用 户 信息 保存 在 Session 中 ， 当 用 
户 访问 每 个 页 面 时 ， 都 检查 Session 以 确认 用 户 身份 ， 如 果 身 份 不 符 则 拒绝 访问 。 由 于 后 
台 管 理 页 面 很 多 ， 如 果 在 每 个 页 面 的 Page Load 事件 中 都 编写 代码 检测 用 户 身份 ， 工 作 量 
大 而 且 代 码 重复 。ASP.NET 提供 了 一 种 HttpModule 的 机 制 可 以 很 好 地 解决 这 个 问题 。 
HttpModule 是 ASP.NET 的 一 种 HTTP 处 理 机 制 ， 当 把 HttpModule 注册 到 网 站 或 者 
Web 应 用 中 以 后 ， 每 当 接 到 一 个 新 的 HTTP 请求 时 ， 都 会 执行 HttpModule 中 的 相应 代码 。 
利用 这 种 机 制 ， 可 以 把 所 有 页 面 公用 的 代码 写 到 HttpModule 中 。 

攀登 者 网 上 书店 项 目 正 是 使 用 了 HttpModule 实现 用 户 身份 验证 。 具体 思路 为 : 每 当 有 
新 的 HTTP 请 求 到 达 时 ， 获 取 所 请 求 URL， 并 分 析 此 URL 所 对 应 页 面 是 否 在 网 上 书店 根 
目录 的 Admin 文件 夹 中 。 如 果 在 Admin 文件 夹 中 ， 则 说 明 用 户 请 求 访问 后 台 管 理 页 面 ， 
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需要 验证 用 户 身份 ， 如 果 不 在 Admin 文件 夹 中 ， 则 说 明 用 户 访问 的 是 普通 页 面 ， 不 需要 身 
份 验证 。 另 外 ，Admin 文件 夹 中 有 两 个 特殊 页 面 AdminLogin 和 Logout 不 需要 身份 验证 。 
代码 如 下 : 


public class CheckAdminModule : IHttpModule 
| 


#region IHttpModule 成 员 
public void Dispose () 
{ } 
public void Init(HttpApplication context) 
{ 
context .AcquireRequestState += new EventHandler (OnRequest); 
} 


#endregion 
public void OnRequest (Object source, EventArgs e) 
! 
Uri url = HttpContext.Current .Request .Url; // 得 到 所 请 求 的 URL 


// 请 求 Rdamin 目录 下 的 文件 时 ， 需 要 进行 身份 验证 ， 只 有 管理 员 才 能 访问 
if (url.AbsolutePath.ToLower() .StartsWith("/admin")) 
{ 
//adminlogin.aspx 和 1logout .aspx 不 需要 身份 验证 
if (url.AbsolutePath.ToLower() .EndsWith ("adminlogin.aspx")) 
return; 
if (url.AbsolutePath.ToLower() .EndsWith ("logout.aspx")) 
return; 
// 如 果 Session 中 没有 保存 管理 员 信息 ， 则 转 到 管理 员 登 录 页 面 
if (HttpContext.Current.Session[Constants.SessionAdmin] == 
null) 
{ 
HttpContext .Current .Response.Redirect ("AdminLogin. 
aspx?url=" 
+HttpContext .Current .Server.HtmlEncode (url. 
PathAndQuery)); 


HttpModule 需要 在 web.config 文件 中 注册 后 才能 起 作用 。 在 web.config 文件 中 添加 以 
下 代码 注册 CheckAdminUser。 
<httpModules> 


<add name="CheckAdminModule" type="BookShop.Admin.CheckAdminModule"/> 
</httpModules> 


4.3.2 ”管理 员 登 录 和 修改 密码 


管理 员 登 录 时 ， 输 入 用 户 名 和 密码 ， 登 录 成 功 后 ， 将 登录 信息 保存 到 Session 中 。 管 
理 登 录 页 面 AdminLogin 代码 如 下 : 
//AdminLogin.aspx 


<form id="forml" runat="server"> 

<div> 

<img src="../images/adminlogin.jpg”alt=" 管 理 员 登录 " style="width:500px;" 
/><br /> 

用 户 名 : <asp:TextBox ID="username" runat="server" /> 


2 


第 1 篇 ASPNET 网 络 开发 关键 技术 


<%-- 验 证 控件 ， 必 须 输入 用 户 名 --%> 
<asp:RequiredFieldValidator ID="RequiredFieldValidatorl" Display="None" 
runat="server"” ErrorMessage=" 必 须 输入 用 户 。" ControlToValidate= 
"username"> 
</asp:RequiredFieldValidator> 
密码 : <asp:TextBox ID="password" runat="server" TextMode="Password" /> 
<%-- 验 证 控件 ， 必 须 输 入 用 户 名 --g%> 
<asp:RequiredFieldValidator 
ID="RequiredFieldValidator2" runat="server" ErrorMessage=" 必 须 输入 密码 。" 
ControlToValidate="password" Display="None"></asp: 
RequiredFieldValidator> 
<asp:Button ID="Buttonl" runat="server"” Text=" 登 录 " onclick= 
"Buttonl Click” /> 
<asp:ValidationSummary ID="ValidationSummaryl" runat="server" 
ForeColor="Red" /> 
</div> 
</form> 
// AdminLogin.aspx.cs 
public partial class AdminLogin : System.Web.UI.Page 
{ 
protected void Page Load(object sender, EventArgs e) 
| We 
protected void Button1l Click(object sender, EventArgs e) 
i 
SqlHelper db = new SqlHelper(); 
// 从 数据 库 验 证 用 户 名 和 密码 是 否 正确 
DbCommand command = db.GetSqlStringCommond( 
"select count (*) from AdminUser where AdminID=@id and Password= 
Q@pass") 7 
db.AddInParameter (command, "@id", DbType.String, username.Text); 
db.AddInParameter (command, "@pass", DbType.String, password.Text); 
int n = Convert.ToInt32 (db.ExecuteScalar (command)); 
// 如 果 用 户 不 存在 则 在 页 面 上 弹出 提示 消息 
if (n == 0) 
{ 
Session[Constants.SessionAdmin] = null; 
Page.ClientScript.RegisterStartupScript (this.GetType (), 
"errorlogin", 


"<script>alert (' 你 输入 的 用 户 名 或 密码 不 正确 。') ;</script>"); 


return; 
} 
// 登 录 成 功 后 ， 将 登录 信息 保存 到 Session 中 ， 并 跳 转 到 所 请 求 的 页 面 
Session[Constants.SessionAdmin] = username.Text; 
if (Request.QueryString["url"] != null) 
Response.Redirect (Server.HtmlDecode (Request. QueryString 
De 
else 


Response.Redirect ("BookAdmin.aspx"); 
} 
在 修改 密码 页 面 AdminChangePass.aspx 中 ,用 户 输入 原 密 码 和 两 次 新 密码 进行 密码 修 


改 。 在 页 面 上 除 3 个 密码 框 外 ， 还 有 3 个 RequiredFieldValidator 验证 控件 保证 输入 密码 不 


为 


空 ， 一 个 CompareValidator 验证 控件 保存 两 次 新 密码 一 致 。 页 面 代码 如 下 : 


//AdminChangePass.aspx 
<form id="forml" runat="server"> 
<div> 
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<h3> 修 改 密码 </h3> 
原 密 码 : <asp:TextBox ID="oldPass" runat="server" TextMode="Password"> 
</asp:TextBox> 
<asp:RequiredFieldValidator ID="RequiredFieldValidatorl" runat= 
"server”" 
ErrorMessage=" 原 密码 不 能 为 空 " ControlToValidate="oldPass"> 
</asp:RequiredFieldValidator> 
br 
新 密码 : <asp:TextBox ID="newPassl" runat="server" TextMode="Password" 
></asp: TextBox> 
<asp:RequiredFieldValidator ID="RequiredFieldValidator2" 
runat="server™" 
ErrorMessage=" 新 密码 不 能 为 空 " ControlToValidate="newPass1l"> 
</asp:RequiredFieldValidator> 


<br /> 
新 密码 : <asp:TextBox ID="newPass2" runat="server" TextMode="Password">< 
/asp:TextBox> 
<asp:RequiredFieldValidator ID="RequiredFieldValidator3" runat= 
"server" 


ErrorMessage=" 新 密码 不 能 为 空 " ControlToValidate="newPass2"> 
</asp:RequiredFieldValidator> 
<asp:CompareValidator ID="CompareValidatorl" runat="server" 
ErrorMessage=" 两 次 输入 新 密码 不 一 致 " ControlToValidate="newPass2" 
ControlToCompare="newPassl"></asp:CompareValidator> 
<br /> 
<asp:Button ID="Button1"” runat="server"” Text=" 修 改 密码 " onclick= 
"Buttonl Click" /> 
<input type="button"” value=" 关 闭 窗 口 " onclick="window.close();" /> 
</div> 
</form> 
//AdminChangePass.aspx.cs 
public partial class AdminChangePass : System.Web.UI.Page 
{ 
protected void Page Load(object sender, EventArgs e) 
. 
protected void Buttonl Click(object sender, EventArgs e) 
{ 
// 获 取 Session 中 保存 的 管理 员 登 录 信息 
string user = Convert.ToString (Session[Constants.SessionAdmin]); 
// 从 数据 库 查询 原 密码 
string sql = "select Password from AdminUser where AdminID=@admin"; 
SqlHelper db = new SqlHelper(); 
System.Data.Common.DbCommand command = db.GetSqlstringCommond (sql); 
db.AddInParameter (command, "@admin",System.Data.DbType.Sstring, 
user); 
string old = db.ExecuteScalar (command) .ToSstring(); 
// 比 较 原 密码 输入 是 否 正确 ， 如 果 不 正确 则 提示 ， 不 设置 新 密码 
if (old != oldPass.Text) 
{ 
ClientScript.RegisterStartupScript (this.GetTYype()，"fail"， 
"<script>alert (' 输 入 的 原始 密码 不 正确 ! ') ;</script>"); 
return; 
// 原 密码 输入 正确 ， 则 修改 新 密码 
sql = "update AdminUser set Password=@pass where AdminID=Q@admin"; 
command = db.GetSqlstringCommond (sql); 
db.AddInParameter (command, "@pass", System.Data.DbType.Sstring, 
newPassl1 .Text); 
db.AddInParameter (command, "@admin", System.Data.DbType.Sstring, 


“ls 


第 1 篇 ASPNET 网 络 开发 关键 技术 


4.3.3 


user); 
db.ExecuteNonQuery (command); 
ClientScript.RegisterStartupScript (this.GetType(), "suceess", 


"<script>alert (' 密 码 修改 成 功 ! ') ;</script>"); 


后 台 管 理 母 版 页 


网 站 后 台 和 前 台 使 用 不 同 的 页 面 布局 和 导航 菜单 ， 所 以 后 台 管 理 页 面 不 能 使 用 前 台 的 
母 版 页 ， 而 是 要 单独 定义 母 版 页 。 后 台 母 版 页 结构 比较 简单 ， 主 要 包括 功能 导航 菜单 。 当 
用 户 在 导航 菜单 中 选择 “修改 密码 ”时 ， 会 调用 JavaScript 代码 弹出 一 个 模式 对 话 杠 。 后 
台 母 版 页 Admin.master 文件 代码 如 下 : 


<head runat="server"> 


<title> 梦 登 者 网 上 书店 </title> 
<script type="text/javascript" language="javascript"> 
function changePassDialog() { 


window.showModalDialog ("AdminChangePass.aspx","","dialogWidth:500px;dia 
logHeight:300px;"); 
} 


</script> 


</head> 
<body> 


<form id="forml" runat="server"> 
<div id="page"> 
<div id="header" style="background-color:#e0f0ff; border:lpx solid 
green; height:40px; "> 
<div style="float:left;"> 
<img src="../images/10go.jpeg" alt=" 梦 登 者 网 上 书店 " style="height:30px; 
vertical-align:bottom;" /> 
<span style="font-size:20px; font-weight:bold;"> 欢 迎 使 用 网 上 书店 后 台 管理 
</span> 
</div> 
<div style="float:right; margin-top:10px;"> 
<a href="BookAdmin.aspx"> 图 书 管理 </a> | 
<a href="CategoryAdmin.aspx"> 类 别管 理 </a> | 
<a href="#"> 会 员 管 理 </a> | 
<a href="#"> 订 单 管理 </a> | 
<a href="#"> 销 售 统计 </a>1 
<a href="javascript:changePassDialog() ;"> 修 改 密码 </a> | 
<a href="../Default.aspx"> 返 回首 页 </a> | 
<a href="Logout .aspx"> 退 出 登录 </a> 
</div> 
</div> 
<div style="clear:both;"> 
<asp:ContentPlaceHolder ID="ContentPlaceHolderl1l" runat="server"> 


</asp:ContentPlaceHolder> 
</div> 

</div> 

<div id="footer" /> 

</form> 


</body> 
</html> 
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后 台 管理 母 版 页 界面 如 图 4.7 所 示 。 


强 ” 欢迎 使 用 网 上 书店 后 台 管 理 图 书 管理 | 类 别管 理 | 会 员 管理 | 订单 管理 | 销售 统计 | 修改 密码 | 返回 首页 | 退出 登录 


4.3.4 


图 4.7 后 台 管 理 母 版 页 


图 书 类 别管 理 


在 图 书 类 别管 理 页 面 CategoryAdmin.aspx 中 ,管理 员 可 以 添加 、 删 除 、 修 改 图 书 类 别 。 
由 于 图 书 类 别 数据 量 少 ， 而 且 使 用 这 个 页 面 的 用 户 少 〈 只 有 管理 员 才 用 ) ， 在 这 个 页 面 中 
可 以 使 用 GridView 和 SqlDataSource 自 带 的 数据 编辑 、 删 除 和 分 页 功能 ， 不 会 影响 应 用 程 


序 性 能 。 


(1) 以 Admin.master 作为 母 版 页 添加 一 个 新 页 面 CategoryAdmin.aspx。 

(2) 在 页 面 上 放置 一 个 GridView 和 SqlDataSource。 配置 SqlDataSource 使 其 读 取 数 据 
库 中 Category 表 的 数据 ， 并 生成 INSERT、DELETE、UPDATE 语句 。 设置 GridView 控件 
的 数据 源 为 SqlDataSource， 并 设置 各 列 标题 ， 启 用 编辑 和 删除 。 

<br /><b> 现 有 图 书 类 别 </b><br /> 


<asp:GridView ID="grid" runat="server" AutoGenerateColumns="False" 


DataKeyNames="CategoryID" DataSourceID="sqlDatal" AllowPaging="True"> 
<Columns> 
<asp:CommandField ShowDeleteButton="True" ShowEditButton="True" 
EditText=" 编 辑 ” UpdateText=" 保 存 " CancelText=" 取 消 ” DeleteText=" 删 除 
" HeaderText=" 操 作 " /> 
<asp:BoundField DataField="CategoryID"” HeaderText=" 类 别 编号 " 
ReadOnly="True" 
SortExpression="CategoryID" ItemStyle-Width="100" ItemStyle- 
HorizontalAlign="Center"/> 
<asp:BoundField DataField="CategoryName"” HeaderText=" 类 别名 称 " 
SortExpression="CategoryName" ItemStyle-Width="300" ItemStyle- 
HorizontalAlign="Center" /> 
</Columns> 


</asp:GridView><br /><br /> 
<asp:SqlDataSource ID="sqlDatal" runat="server" ConnectionString="<%$ 
Connectionstrings:database%®>" 


DeleteCommand="DELETE FROM [Category] WHERE [CategoryID] = @CategoryID" 
InsertCommand="INSERT INTO [Category] ([CategoryName]) VALUES 
(@CategoryName)" 
ProviderName="<%$ Connectionstrings:database.ProviderName $%>" 
SelectCommand="SELECT [CategoryID], [CategoryName] FROM [Category]" 
UpdateCommand="UPDATE [Category] SET [CategoryName] = @CategoryName 
WHERE [CategoryID] = @CategoryID"> 
<DeleteParameters> 

<asp:Parameter Name="CategoryID" Type="Int32" /> 
</DeleteParameters> 
<InsertParameters> 

<asp:Parameter Name="CategoryName" Type="String" /> 
</InsertParameters> 
<UpdateParameters> 

<asp:Parameter Name="CategoryName" Type="String" /> 

<asp:Parameter Name="CategoryID" Type="Int32" /> 
</UpdateParameters> 


</asp:SqlDataSource> 
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(3) 在 页 面 上 添加 一 个 DetailsView 控件 以 添加 新 的 图 书 类 别 。 设 置 DetailsView 的 数 
据 源 为 SqlDataSource， 设 置 睦 认 模 式 为 插入 模式 。 


<b> 添 加 新 类 别 </b><br /> 

<asp:DetailsView ID="DetailsViewl" runat="server" DataSourceID="sqlDatal™" 
DefaultMode="Insert" AutoGenerateRows="False" > 

<Fields> 
<asp:BoundField DataField="CategoryName"” HeaderText=" 类 别名 称 " /> 
<asp:CommandField ShowInsertButton="True"” InsertText=" 添 加 " 
ShowCancelButton="false" /> 

</Fields> 

</asp:DetailsView> 


(4) CategoryAdmin.aspx 页 面 运行 界面 如 图 4.8 
所 示 。 
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4.3.5 图 书 管理 


在 图 书 管理 页 面 中 ， 管 理 员 可 以 搜索 和 查看 图 书 
信息 ， 并 可 以 单 击 相应 链接 对 图 书 进 行 编辑 和 删除 操 
作 。 这 个 页 面 与 前 台中 的 高 级 搜索 页 面 有 一 些 相同 之 
处 ， 但 是 二 者 在 图 书 列表 中 对 图 书 的 可 用 操作 不 同 ， 
前 台 可 以 将 图 书 添 加 到 购物 车 和 查看 图 书 详情 ， 而 后 
台 则 可 以 删除 图 书 和 编辑 图 书信 息 。 

(1) 以 Admin.master 作为 母 版 页 添加 一 个 新 页 面 
BookAdmin.aspx。 

(2) 在 BookAdmin.aspx 页 面 中 放置 多 个 TextBox 
控件 以 输入 搜索 条 件 。 图 4.8 图 书 类 别管 理 界面 

<div class="dashedborder"> 

标题 : <asp:TextBox ID="bookTitle" runat="server"/> 

&nbsp; 作 者 : <asp:TextBox ID="author" runat="server"/> 

&nbsp; 出 版 社 : <asp:TextBox ID="publisher" runat="server"/> 

<a style="margin-left:150px; text-decoration:underline; "href= 

"BookEdit .aspx?id=-1"> 添 加 新 书 </a><br /> 

价格 范围 : <asp:TextBox ID="pricel" runat="server"></asp:TextBox> 至 <asp: 


TextBox ID="price2" runat="server"></asp:TextBox> 
&nbsp; 


出 版 日 期 <asp:TextBox ID="datel" runat="server"></asp:TextBox> 至 <asp: 
TextBox ID="date2" runat="server"></asp:TextBox> 


&nbsp; <asp:Button ID="Buttonl" runat="server"” Text=" 搜 索 " Width="80" 
onclick="Buttonl Click" /> 
</div> 


(3) 在 BookAdmin.aspx 页 面 中 放置 一 个 Repeater 控件 和 一 个 AspNetPager 控件 以 分 页 
形式 显示 图 书信 息 。 在 Repeater 控件 中 除 显示 图 书信 息 外 ， 还 有 两 个 链接 : 删除 图 书 和 编 
辑 图 书 。 


<asp:Repeater ID="Repeaterl" runat="server" onitemcommand="Repeaterl 
ItemCommand" > 
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<HeaderTemplate><table></HeaderTemplate> 

<ItemTemplate> 

区 蕊 天 之 

<td> 

<%-- 将 图 书 封面 作为 一 个 超 链接 --%> 

<a href="BookEdit.aspx?id=<%#Eval ("BookID")$%>"> <img src="../BookImages/< 
S$#Eval ("SmallImage") %>" alt="<%#Eval ("BookTitle") %>" style="width: 
100px;" /></a> 


</td> 
<td> 
<h3> <%#Eval ("BookTitle") $> </h3> <s- -图书 标题 --g%> 


<%-- 以 下 内 容 显 示 了 图 书 的 作者 、 定 价 、 会 员 价 、 折 扣 信息 --g%> 

作者 : <#s#Convert .ToString (Eval ("Author1")) + Convert.ToString (Eval 
("Author2")) + Convert.ToString (Eval ("Author3"))%>| 

出 版 社 :，<%#Eval ("PublisherName") %>| 出 版 日 期 <%$#Eval ("PublishDate") %> 

<br /><br /> 

定价 : <span style="color:Blue;"> <%#Eval ("Price", "{0:C}") $></span> | 

会 员 价 : <span style="color:red;"><%#Eval ("RealPrice", "{0:C}")%></span> 
折扣 : <%#Eval ("Discount")%><br /><br /> 

< 一 -以 下 为 修改 图 书 和 删除 图 书 两 个 链接 --%> 

<span style="margin-left:50px;"> 

<a href="BookEdit.aspx?id=<%#Eval ("BookID") s>"> 修 改 图 书信 息 </a> 

</span> 

<asp:LinkButton ID="Buttonl" runat="server" Text=" 删 除 图 书 ” CommandName= 

"delete" 

CommandArgument="<%#Eval ("BookID") %>' OnClientClick="return 

confirmDelete (this);" /> 

</td> 

</tr> 

</ItemTemplate> 

<FooterTemplate></table></FooterTemplate> 

</asp:Repeater> 

<%-- 分 页 控件 --%> 

<webdiyer:AspNetPager ID="AspNetPager]l" runat="server" 

onpagechanged="AspNetPagerl] PageChanged"> 
</webdiyer:AspNetPager> 


(4) 在 “搜索 ”按钮 的 Click 事件 中 及 AspNetPager 控件 的 PageChanged 事件 中 ， 编 
写 代码 实现 搜索 结果 的 分 页 显示 。 具 体 代码 与 前 台 的 高 级 搜索 页 面 类 似 ， 此 处 省 略 。 
(5) 在 Repeater 控件 的 IemCommand 事件 中 ， 编 写 代码 删除 所 选择 的 图 书 。 


protected void Repeaterl ItemCommand(object source, RepeaterCommand-— 
EventArgs e) 
{ 
if (e.CommandName == "delete") // 如 果 命 令 名 为 删除 
int id = Convert.ToInt32 (e.CommandArgument);// 获 取 命 令 参数 (图 书 ID) 
// 构 建 delete 语句 ， 并 执行 
string sql = "delete from Book where BookID=@id"; 
SqlHelper db = new SqlHelper(); 
Var Command = db.GetSqlstringCommond (sql); 
db.AddInParameter (Command, "@id", System.Data.DbType.Int32, id); 
db.ExecuteNonQuery (Command); 


bindData(); // 重 新 绑 定数 据 


} 
(6) BookAdmin.aspx 页 面 运行 界面 如 图 4.9 所 示 。 
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图 4.9 图 书 管理 页 面 


4.3.6 图 书 详情 编辑 设计 思 


在 图 书 详情 编辑 页 面 中 ， 管 理 员 可 以 修改 图 书 的 所 有 明细 信息 ， 如 标题 、 作 者 、 目 录 、 
内 容 简介 、 封 面 等 。 图 书 详情 涉及 的 信息 很 多 ， 而 且 数 据 量 较 大 ， 例 如 ， 图 书 的 目录 和 内 
容 简介 都 会 包含 许多 文字 ， 图 书 的 大 封面 和 小 封面 图 片 也 会 占据 页 面 很 大 的 空间 ， 在 一 个 
页 面 上 不 足以 显示 全 部 图 书信 息 。 

为 了 显示 和 编辑 图 书信 息 的 方便 ， 可 以 使 用 类 似 于 标签 页 控件 Tab 的 控件 组 成 图 书信 
息 。 把 整个 图 书信 息 拆 分 成 几 个 部 分 ， 在 页 面 上 任 一 时 刻 只 显示 其 中 某 一 部 分 内 容 ， 可 以 
方便 地 在 几 个 部 分 之 间 进 行 切换 。 在 ASP.NET 标准 服务 器 控件 中 不 包含 Tab 控件 ， 但 是 
提供 了 一 个 类 似 的 控件 叫做 MultiView。 

入 提示 : 网 上 有 许多 第 三 方 控件 库 实 现 了 Tab 控件 ， 也 可 以 使 用 这 些 第 三 方 控件 完成 图 书 

详情 编辑 页 面 的 布局 。 

正如 其 名 称 所 示 ，MultiView 控件 可 以 在 一 个 页 面 上 显示 多 个 视图 。MultiView 中 包含 
多 个 视图 ， 每 个 视图 都 有 一 个 索引 ， 任 一 时 刻 只 有 一 个 视图 可 见 ， 这 个 可 见 视图 称 为 活动 
视图 。 可 以 通过 设置 MultiView 控件 的 ActiveViewIndex 属性 在 各 个 视图 之 间 进 行 切换 。 

于 图 书 详情 包含 的 数据 量 太 大 ， 如 果 把 所 有 图 书信 息 放 在 同一 个 页 面 中 ， 即 使 使 用 
了 MultiView 控件 ， 页 面 代码 也 会 变 得 见长 、 上 涩 、 不 易 维护 。 为 了 缩小 模块 规模 ， 可 以 
把 图 书信 息 拆 分 成 几 个 部 分 ， 每 个 部 分 分 别 用 一 个 用 户 控件 显示 和 编辑 。 按 照 这 种 思路 ， 
可 以 把 图 书信 息 拆 分 成 3 部 分 : 图 书 基本 信息 、 图 书 封面 图 片 和 图 书 所 属 类 别 ， 并 且 分 别 
用 3 个 用 户 控件 完成 各 个 部 分 信息 的 显示 和 编辑 。 


4.3.7 图书 基本 信息 编辑 控件 


在 图 书 基本 信息 控件 中 ， 可 以 显示 和 修改 一 个 特定 图 书 除 封面 图 片 和 类 别 以 外 的 所 有 
信息 。 在 这 些 信息 中 有 两 个 属性 比较 特殊 : 图 书目 录 和 内 容 简介 ， 这 两 个 属性 包含 文字 量 
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较 大 ， 而 且 希 望 包含 格式 控件 符号 ， 从 而 可 以 设置 字体 、 颜 色 、 边 框 等 。 可 以 使 用 HIML 
编辑 器 来 满足 这 种 需求 。HTML 编辑 器 界面 就 像 一 个 精简 版 的 Word 软件 ， 允 许 用 户 以 所 


见 即 所 得 和 


本 项 目 使 


的 方式 编辑 文字 ， 并 自动 生成 对 应 的 HTML 代码 。 网 上 有 多 种 HTML 编辑 器 ， 
FreeTextBox 控件 〈 控 件 网 址 为 http:/freetextbox.com/) 。 


(1) 添加 一 个 BookEditControl 用 户 控件 。 

(2) 下 载 FreeTextBox 控件 并 将 其 添加 到 Visual Studio 工具 箱 中 。 

(3) 在 BookEditControl 用 户 控件 中 添加 一 个 SqlDataSource 控件 ， 从 Book 表 读 取 数 
据 ， 并 生成 UPDATE 和 DELETE 语句 。 要 注意 从 代码 视图 中 手工 删除 SqlDataSource 控件 
自动 生成 的 Update 语句 中 设置 SmallImage 和 BigImage 的 代码 。 


<asp:SqlDataSource ID="bookDataSource" runat="server" 
Connectionstring="<%$ ConnectionStrings:database $%>" 


SelectCommand="SELECT * FROM [Book] WHERE ([BookID] = @BookID)" 
DeleteCommand="DELETE FROM [Book] WHERE [BookID] = QBookID" 
InsertCommand="INSERT INTO [Book] ([BookTitle], [Authorl], [Author2], 


[Author3], 


[PublisherID], 


[Price], [Discount], [PublishDate], 


[Description], [Contents], [SaleSum], [ClickCount]) VALUES (@BookTitle, 
@Authorl, @Author2, @Author3, @PublisherID, @Price, @Discount, 
@PublishDate, @Description, @Contents, @SaleSum, @ClickCount)" 


UpdateCommand="UPDATE [Book] SET [BookTitle] = @BookTitle, [Authorl] 
@Authorl, [Author2] = @Author2, [Author3] = @Author3, [PublisherID] 
@PublisherID, [Price] = @Price, [Discount] = @Discount, [PublishDate] 
= @PublishDate, [Description] = @Description, [Contents] = @Contents, 
[SaleSum] = @SaleSum, [ClickCount] = @ClickCount WHERE [BookID] = 
@BookID"> 
<DeleteParameters> 

<asp:Parameter Name="BookID" Type="Int32" /> 
</DeleteParameters> 
<InsertParameters> 


<asp:Parameter 
<asp:Parameter 
<asp:Parameter 
<asp:Parameter 
<asp:Parameter 
<asp:Parameter 
<asp:Parameter 
<asp:Parameter 
<asp:Parameter 
<asp:Parameter 
<asp:Parameter 
<asp:Parameter 


Name="BookTitle" Type="String" /> 
Name="Authorl" Type="String" /> 
Name="Author2" Type="String" /> 
Name="Author3" Type="String" /> 
PublisherID" Type="Int32" /> 
Price" Type="Double" /> 
Discount" Type="Double" /> 
PublishDate" Type="DateTime" /> 
Name="Description" Type="String" /> 
Name="Contents" Type="String" /> 
Name="SaleSum" Type="Int32" /> 
Name="ClickCount" Type="Int32" /> 


</InsertParameters> 


<SelectParameters> 


<UpdateParameters> 


<asp:QueryStringParameter Name="BookID" QueryStringField="id" 
Type="Int32" /> 
</SelectParameters> 


<asp:Parameter 
<asp:Parameter 
<asp:Parameter 
<asp:Parameter 
<asp:Parameter 
<asp:Parameter 
<asp:Parameter 
<asp:Parameter 
<asp:Parameter 


Name="BookTitle" Type="String" /> 
Name="Authorl" Type="String" /> 
Name="Author2" Type="String" /> 
Name="Author3" Type="String" /> 
Name="PublisherID" Type="Int32" /> 
Name="Price" Type="Double" /> 
Name="Discount" Type="Double" /> 

Nam PublishDate" Type="DateTime" /> 
Name="Description" Type="String" /> 
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<asp:Parameter Name="Contents" Type="String" /> 
<asp:Parameter Name="SaleSum" Type="Int32" /> 
<asp:Parameter Name="ClickCount" Type="Int32" /> 
<asp:Parameter Name="BookID" Type="Int32" /> 
</UpdateParameters> 
</asp:SqlDataSource> 


全 提示 : 由 于 此 控件 不 修改 图 书 封面 图 片 ， 所 以 要 从 代码 视图 中 手工 删除 SqlDataSource 
控件 自动 生成 的 Update 语句 中 设置 SmallImage 和 BigImage 的 代码 , 否则 在 更 新 
数据 时 就 会 把 封面 图 片 清空 。 


(4) 在 BookEditControl 用 户 控件 中 添加 一 个 FormView 控件 ， 配 置 FommView 的 
ItemTemplate 以 显示 和 编辑 图 书信 息 ， 注 意 其 中 使 用 FreeTextBox 控件 编辑 图 书简 介 和 图 
书目 录 。 


<asp:FormView ID="FormViewl" runat="server" DataKeyNames="BookID" 


DataSourceID="bookDataSource" DefaultMode="Edit" 
oniteminserted="FormViewl]l ItemInserted"> 
<EditItemTemplate> 
图 书 编号 : 


<asp:Label ID="BookIDLabell" runat="server" Text="'<%# Eval 
("BookID") $>' /> 
<br /> 
题目 :<asp:TextBox ID="BookTitleTextBox" runat="server" Text='<%# 
Bind("BookTitle") %>' Width="200" /> 
作者 1:<asp:TextBox ID="AuthorlTextBox" runat="server" Text='<%# 
Bind("Authorl1") %>' /> 
作者 2:<asp:TextBox ID="Author2TextBox" runat="server" Text="'<%# 
Bind ("Author2") %>' /> 
作者 3: <asp:TextBox ID="Author3TextBox" runat="server" Text="'<%# 
Bind ("Author3") $%$>' /> 
<br /> 
出 版 社 : 
<asp:DropDownList runat="server" ID="publisherList" Width="200" 
DataValueField="PublisherID" DataTextField="PublisherName" 
SelectedValue='<%# Bind("PublisherID") %>' DataSourceID="public- 
sherDataSource" /> 
定价 :<asp:TextBox ID="PriceTextBox" runat="server" Text='<%# 
Bind("Price") %>°' /> 
折扣 :<asp:TextBox ID="DiscountTextBox" runat="server" Text='<%# 
Bind("Discount") %>' /> 
会 员 价格 : <asp:Label ID="RealPriceTextBox" runat="server" Text='<%# 
Eval ("RealPrice") %>' /> 
<be /> 
出 版 日 期 : <asp:TextBox ID="PublishDateTextBox" runat="server" Text= 
'<%# Bind("PublishDate") %>' /> 
总 销售 量 : <asp:TextBox ID="SaleSumTextBox" runat="server" Text='<%# 
Bind("SaleSum") %$>' /> 
总 单 击 量 : <asp:TextBox ID="ClickCountTextBox" runat="server" Text= 
'<$%# Bind("ClickCount") %>' /> 
<asp:Button ID="UpdateButton" runat="server" CausesValidation="True" 
CommandName="Update" Text=" 保 存 "” /> 
&nbsp; <asp:Button ID="UpdateCancelButton" runat="server" 
CausesValidation="False" CommandName="Cancel" Text=" 取 消 " /> 
“br > 
图 书简 介 :<br /> 
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<FTB:FreeTextBox ID="descriptionTextBox" runat="server" Text="'<$# 
Bind("Description") $>" 
Width="900" Height="200"> </FTB:FreeTextBox> <br /> 
目录 :<br /> 
<FTB:FreeTextBox ID="ContentTextBox" runat="server" Text="'<%# 
Bind("Contents") $%$>" 
Width="900" Height="200"> </FTB:FreeTextBox><br /> 
</EditItemTemplate> 
<InsertItemTemplate> 
图 书 编号 : 
<asp:Label ID="BookIDLabe11" runat="server" Text='<g# Eval ("Book-- 
DD 
<br /> 
题目 :<asp:TextBox ID="BookTitleTextBox" runat="server" Text="'<%# 
Bind("BookTitle") %>' Width="200" /> 
作者 1:<asp:TextBox ID="AuthorlTextBox" runat="server" Text='<%# 
Bind("Authorl1") %>"' /> 
作者 2:<asp:TextBox ID="Author2TextBox" runat="server" Text='<%# 
Bind("Author2") %>"' /> 
作者 3: <asp:TextBox ID="Author3TextBox" runat="server"” Text='<%# 
Bind("Author3") %>' /> 
<br /> 
出 版 社 : 
<asp:DropDownList runat="server" ID="publisherList" Width="200" 
DataValueField="PublisherID" DataTextField="PublisherName" 
SelectedValue='<%# Bind("PublisherID") %>' DataSourceID="public- 
SherDataSource" /> 
定价 :<asp:TextBox ID="PriceTextBox" runat="server" Text='<%# Bind 
("Price") %>' /> 
折扣 :<asp:TextBox ID="DiscountTextBox" runat="server" Text='<%# 
Bind("Discount") %>' /> 
<br /> 
出 版 日 期 : <asp:TextBox ID="PublishDateTextBox" runat="server" Text= 
'<%# Bind("PublishDate") %>' /> 
总 销售 量 : <asp:TextBox ID="SaleSumTextBox" runat="server" Text='<%# 
Bind("SaleSum") $%>' /> 
总 单 击 量 : <asp:TextBox ID="ClickCountTextBox" runat="server" Text= 
'<%# Bind("ClickCount") %>' /> 
<asp:Button ID="UpdateButton" runat="server" CausesValidation= 
"True" 
CommandName="Insert"” Text=" 添 加 ” /> 
&nbsp; <asp:Button ID="UpdateCancelButton" runat="server" 
CausesValidation="False" CommandName="Cancel"” Text=" 取 消 " /> 
<br /> 
图 书简 介 :<br /> 
<FTB:FreeTextBox ID="descriptionTextBox" runat="server" Text="'<%# 
Bind("Description") g>" 
Width="900" Height="200"> </FTB:FreeTextBox> <br /> 
目录 :<br /> 
<FTB:FreeTextBox ID="ContentTextBox" runat="server" Text='<%# Bind 
("Contents") $>" 
Width="900" Height="200"> </FTB:FreeTextBox><br /> 
</InsertItemTemplate> 
</asp:FormView> 


(5) 在 BookEditControl 用 户 控件 中 可 放置 一 个 SqlDataSource 控件 ， 从 Publisher 表 中 
检索 数据 ， 并 绑 定 到 FormView 控件 IemTemplate 中 的 DropDownList 控件 。 
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<asp:SqlDataSource ID="publisherDataSource" runat="server" 
ConnectionString="<%$ ConnectionStrings:database %>" 
SelectCommand="SELECT [PublisherID], [PublisherName] FROM [Publisher] 
"></asp:SqlDataSource> 


(6) 在 BookEditControl 用 户 控 件 的 Page Load 事件 中 ， 根 据 QueryString 传递 的 图 书 
编号 设置 FormView 的 格式 。 如 果 图 书 编号 为 大 于 0， 则 以 编辑 模式 显示 此 编号 对 应 的 图 
书 ， 如 果 图 书 编号 为 -1， 则 说 明 需 要 添加 新 图 书 ， 将 FormView 更 改 为 插入 模式 。 


protected void Page Load (object sender, EventArgs e) 


有 
if (!IsPostBack) 


{ 
if (Request.QueryString["id"] == "-1") 
FormViewl .ChangeMode (FormViewMode.Insert); 


(7) 当 新 图 书信 息 首次 保存 时 ， 将 页 面 跳 转 到 BookEdit 页 面 ， 并 将 新 图 书 的 ID 作为 
QueryString 参数 传递 过 去 。 为 此 ， 需 要 在 FormView 的 ItemInserted 事件 中 编写 以 下 代码 。 


protected void FormView!l ItemInserted(object sender, FormViewInserted-— 
EventArgs e) 


| 
string sql = "select max(BookID) from Book"; // 得 到 最 新 图 书 ID 
SqlHelper db = new SqlHelper(); 
Var command = db.GetSqlStringCommond (sql); 
int n = Convert.ToInt32 (db.ExecuteScalar (command)); 
Response.Redirect ("BookEdit .aspx?id="+n); // 重 新 跳 转 到 此 页 面 


} 
(8) 添加 一 个 新 页 面 以 测试 BookEditControl 用 户 控 件 ， 运 行 界面 如 图 4.10 所 示 。 


人 作者 3 
会 只 价格 85.75 


受 好 评 的 名 作 ， 作 者 | 
a 惟 导 图 “指明 本 音 要 站 
六 都 对 读者 理解 _C#ls 言 有 所 神 益 ， 区 泥 关 型 、 
反 和 还 办 绍 了 请 言 集成 查询 LINQ》 技术 ， 以 及 与 其 居 的 六 展 方法 、 分 部 方法 


、 反 射 ， 如 
Lanbda 表 达 式 、 标 闪 查 询 灌 和 表达 式 等 办 
本 书 适合 对 6# 感 兴趣 的 各 个 屋 次 的 读者 ， 无 论 对 初学 考证 是 经验 的 开发 者 ， 本 书 都 是 一 本 很 有 价值 的 参考 书 。 


2 类 型 定 义 4 


图 4.10 图 书 基本 信息 编辑 控件 


4.3.8 图 书 封面 编辑 控件 


在 图 书 封面 控件 中 ， 可 以 显示 一 个 特定 图 书 的 现 有 的 小 封面 图 片 和 大 封面 图 片 ， 可 以 


“De 


第 4 章 ”阶段 项 目 案例 : 网 上 书店 


上 传 新 的 图 书 封面 替换 原 有 的 封面 。 在 上 传 文件 时 ， 要 注意 把 文件 重 命名 。 这 是 因为 数据 
库 中 会 有 许多 书 ， 如 果 按 照 上 传 时 的 文件 名 保存 文件 ， 则 很 可 能 造成 文件 名 重 名 ， 从 而 覆 
瘟 早 期 的 文件 。 在 文件 上 传 时 ， 应 该 按照 一 种 规则 将 文件 重 命名 ， 以 确保 每 个 文件 名 的 唯 
一 性 。 通常 用 的 重 命名 规则 包括 用 数据 库 主键 、 当 前 日 期 时 间 , 或 者 二 者 结合 作为 文件 名 。 

(1) 添加 一 个 新 用 户 控件 BookCoverEdit.ascx。 

(2) 在 BookCoverEdit 控件 上 放置 一 个 FormView 和 SqlDataSource， 用 于 检索 图 书 封 
面 信息 并 显示 。 图 书 编号 通过 QueryString 传递 给 页 面 ， 要 给 SqlDataSource 添加 一 个 
QueryString 参数 。 


<h3> 图 书 封面 </h3> 

<div style="float:left;"> 

<h3> 原 有 封面 </h3> 

<asp:FormView ID="FormView1" runat="server" DataSourceID="SqlDataSourcel" > 

<ItemTemplate> 

小 封面 <br /> 

<img src="../BookImages/<%#Eval ("Smal1Image")gs>"” style="width:100px;" 

alt="<%#Eval ("BookTitle")®%>"/><br /> 

大 封面 <br /> 

<img src="../BookImages/<%#Eval ("BigImage")®%>" style="width:200px;" alt= 

"<%#Eval ("BookTitle")®%>" /><br /> 

</ItemTemplate> 

</asp:FormView> 

</div> 

<asp:SqlDataSource ID="SqlDataSourcel" runat="server" 
Connectionstring="<%$ Connectionstrings:database %>" 
SelectCommand="SELECT [BookID], [BookTitle], [BigImage], [SmallImage] 
FROM [Book] WHERE ([BookID] = @BookID)"> 
<SelectParameters> 

<asp:QueryStringParameter Name="BookID" QueryStringField="id" Type= 
i Vries rb 这 > 

</SelectParameters> 

</asp:SqlDataSource> 


(3) 在 BookCoverEdit 控件 上 放置 两 个 FileUpload 控件 和 两 个 Button 控件 ， 以 上 传 新 
的 图 书 封面 。 
<div > 
<h3> 上 传 新 封面 </h3> 
小 封面 <asp:FileUpload ID="smallPicture" runat="server" /> 
<asp:Button ID="Buttonl"” runat="server"” Text=" 上 传 并 替换 " onclick= 


"Buttonl Click" /> 
=</ 
大 封面 <asp:FileUpload ID="bigPicture" runat="server" /> 
<asp:Button ID="Button2" runat="server" Text=" 上 传 并 替换 " onclick="Button2 
CeK /DS<br /> 
</div> 


(4) 在 Page Load 事件 中 ， 根 据 QueryString 所 传递 的 图 书 ID 的 合法 性 ， 设 置 两 个 上 
传 按钮 的 可 用 性 。 


protected void Page_Load (object sender, EventArgs e) 
{ 


if (!IsPostBack) 
{ 
string id = Request.QueryString["id"]; 


~ 
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由 


if (string.IsNullOrEmpty(id) 11 id == "-1") 
外 

Button1-Enabled = false; 

Button2 .Enabled = false; 


(5) 在 两 个 上 传 按钮 的 Click 事件 中 ， 将 新 的 封面 图 片 文件 上 传 到 服务 器 ， 并 删除 原 


有 图 片 文件 。 在 上 传 文件 时 ， 把 文件 进行 如 


EE 命名 ， 命 名 规则 为 : 图 书 主 键 +s〈 表 示 small) 


或 b (表示 big) + 扩展 名 。 


protected void Button1l Click(object sender，EventRrgs e) 


{ 


. 


updateBookCover (smallPicture, false); 


protected void Button2 Click(object sender, EventArgs e) 


{ 


1 


updateBookCover (bigPicture, true); 


/// <summary> 

/// 更 新 图 书 封面 

/// </summary> 

/// <param name="file"> 包 含 封面 图 片 的 文件 上 传 控件 </param> 
/// <param name="big"> 是 否 大 封面 </param> 

private void updateBookCover (FileUpload file, bool big) 


{ 


. 194 . 


if (!file.HasFile) return; 
// 取 得 原来 的 封面 文件 并 删除 
SqlHelper db = new SqlHelper(); 
string sql = string.Format ("select {0} from Book where BookID={1}" 
, big?"BigImage":"SmallImage" ,Request.QueryString["id"]); 
DbCommand command = db.GetSqlStringCommond (sql); 
string old = db.ExecuteScalar (command) .ToString() 7 
old = Server.MapPath("~/BookImages/" + old) ; 
if(System.IO.File.Exists(old)) 
System.IO.File.Delete(old); 
// 得 到 新 封面 文件 名 , 命名 规则 为 : 图 书 ID + s 或 b + 文件 扩展 名 
string newName = Request.Querystring["id"] 
BL De 
+ System.IO.Path.GetExtension (file.FileName); 
file.SaveAs (Server.MapPath ("~/BookImages/" + newName)); 
// 更 新 数据 库 中 图 书 封面 数据 
sql="update Book set " + (big?"BigImage":"SmallImage") 
+"=@img where BookID=@id"; 
command = db.GetSqlstringCommond (sql); 
db.AddInParameter (command, "@id", DbType.Int32, 
int.Parse (Request .QueryString["id"])); 
db.AddInParameter (command, "@img", DbType.String, newName); 
db.ExecuteNonQuery (command); 
FormViewl .DataBind(); 


第 4 章 阶段 项 目 案例 : 网 上 书店 


(6) 创建 一 个 测试 页 面 以 测试 BookCoverEdit 控件 ， 运 行 界面 如 图 4.11 所 示 。 
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图 4.11 图 书 封面 编辑 控件 


4.3.9 图 书 类 别 编辑 控件 


在 图 书 类 别 编辑 控件 中 ， 管 理 员 可 以 编辑 一 个 特定 图 书 的 所 属 类 别 。 一 个 图 书 可 以 同 
时 属于 零 个 或 者 多 个 类 别 ， 图 书 和 类 别 之 间 是 多 对 多 的 关系 。 由 于 图 书 类 别 涉及 的 数据 量 
很 小 ， 使 用 这 个 控件 的 用 户 很 少 (只 有 管理 员 才 能 用 ) ， 所 以 这 个 页 面 可 以 使 用 GridView 
自 带 的 数据 编辑 和 删除 功能 。 

(1) 添加 一 个 用 户 控件 BookCategoryControl.ascx。 

(2) 在 用 户 控件 上 放置 一 个 SqlDataSource 控件 ， 从 检索 BookCategory 表 中 检索 数据 ， 
并 自动 生成 UPDATE 和 DELETE 语句 。 注 意 为 SqlDataSource 控件 添加 QueryString 参数 。 


<asp:SqlDataSource ID="SqlDataSourcel" runat="server" 
Connectionstring="<%$ ConnectionStrings:database %>" 
DeleteCommand="DELETE FROM [BookCategory] WHERE [BookID] = @BookID AND 
[CategoryID] = @CategoryID" 
InsertCommand="INSERT INTO [BookCategory] ([BookID], [CategoryID]) 
VALUES (@BookID, Q@CategoryID)" 
SelectCommand="SELECT [BookID], [CategoryID] FROM [BookCategory] WHERE 
([BookID] = @BookID)"> 
<DeleteParameters> 
<asp:Parameter Name="BookID" Type="Int32" /> 
<asp:Parameter Name="CategoryID" Type="Int32" /> 
</DeleteParameters> 
<InsertParameters> 
<asp:Parameter Name="BookID" Type="Int32" /> 
<asp:Parameter Name="CategoryID" Type="Int32" /> 
</InsertParameters> 
<SelectParameters> 
<asp:QueryStringParameter Name="BookID" QueryStringField="id" 
Type="Int32" /> 
</SelectParameters> 
</asp:SqlDataSource> 
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(3) 在 用 户 控件 上 添加 一 个 GridView 控件 ， 将 其 数据 源 设置 为 SqlDataSourcel， 设 置 
GridView 中 的 各 个 字段 标题 .将 GridView 的 所 属 类 别 一 列 设置 为 模板 列 ,用 DropDownList 
控件 显示 图 书 类 别 。 

<h3> 图 书 所 属 类 别 </h3> 


<asp:GridView ID="GridViewl" runat="server" AutoGenerateColumns="False" 
DataKeyNames="BookID,CategoryID" DataSourceID="SqlDataSourcel"> 
<Columns> 
<asp:CommandField ShowDeleteButton="True"” DeleteText=" 删 除 "” /> 
<asp:BoundField DataField="BookID" HeaderText=" 图 书 编号 " ReadOnly= 
"True™ 
ItemStyle-Width="100" ItemStyle-HorizontalAlign="Center" /> 
<asp:TemplateField ItemStyle-Width="300" ItemStyle-Horizon-— 
talAlign="Center"” HeaderText=" 所 属 类 别 "” > 
<ItemTemplate > 
<asp:DropDownList runat="server" ID="categoryList" DataSour- 
ceID="SqlDataSource2" 
DataValueField="categoryID" DataTextField="CategoryName" 
SelectedValue='<%#Eval ("CategoryID") %>' /> 
</ItemTemplate> 
</asp:TemplateField> 
</Columns> 
</asp:GridView> 


(4) 在 用 户 控 件 中 添加 另外 一 个 SqlDataSource, 从 Category 表 中 检索 数据 ,为 GridView 
控件 模板 列 中 的 DropDownList 控件 提供 数据 源 。 


<asp:SqlDataSource ID="SqlDataSource2" runat="server" 
ConnectionSstring="<%$ ConnectionStrings:database %>" 
SelectCommand="SELECT * FROM [Category]"></asp:SqlDataSource> 


(5) 在 用 户 控 件 底部 放置 一 个 DropDownList 和 一 个 Button， 以 添加 新 的 类 别 。 
<b> 添 加 新 的 类 别 </b><br /> 


<asp:DropDownList runat="server" ID="categoryList" DataSourceID= 
"SqlDataSource2" 
DataValueField="categoryID" DataTextField="CategoryName" /> 
<asp:Button ID="buttonl" runat="server" Text=" 添 加 " Width="80" 
onclick="buttonl Click" /> 


(6) 在 “添加 ”按钮 的 Click 事件 中 完成 数据 库 操作 ， 为 图 书 添加 所 选 类 别 。 


protected void button1l Click(object sender, EventArgs e) 
:| 


string bookid = Request.QueryString["id"]; 

string category = categoryList.SelectedValue; 

string sql = "insert into BookCategory (BookID,CategoryID) 

values (@bid, @cid)"; 

SqlHelper db = new SqlHelper(); 

System.Data.Common.DbCommand command=db.GetSqlStringCommond (sql); 
db.AddInParameter (command, "@bid", System.Data.DbType.Int32, bookid); 
db.AddInParameter (command, "@cid", System.Data.DbType.Int32, int.Parse 
(category)); 

db.ExecuteNonQuery (command); 

GridViewl .DataBind(); 
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(7) 添加 一 个 测试 页 面 以 测试 BookCategoryControl 用 户 控 件 ， 运 行 界面 如 图 4.12 
所 示 。 
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图 4.12 图 书 类 别 编辑 用 户 控件 


4.3.10 ”图 书 编辑 页 面 


在 图 书 编辑 页 面 BookEdit.aspx 中 , 以 MultiView 控件 在 3 个 视图 中 显示 前 面 所 创建 的 
3 个 用 户 控件 : 基本 信息 编辑 控件 、 图 书 封面 编辑 控件 、 图 书 类 别 编辑 控件 。 用 户 可 以 在 3 
个 视图 之 间 进 行 切换 ， 查 看 并 编辑 相关 信息 。BookEdit.aspx 页 面 代 码 如 下 工 : 


//BookEdit .aspx 
<%-- 注 册 1 个 自 定义 控件 (HTML 文档 编辑 器 ) 和 3 个 用 户 控件 --%> 
<%@ Register Assembly="FreeTextBox" Namespace="FreeTextBoxControls" 
TagPrefix="FTB" $%> 
<%@ Register src="BookEditControl.ascx" tagname="BookEditControl" 
tagprefix="ucl" 区 > 
<%@ Register src="BookCoverEdit.ascx" tagname="BookCoverEdit" 
tagprefix="uc2" %> 
<%@ Register src="BookCategoryControl .ascx" tagname="BookCategoryControl" 
tagprefix="uc3" 和 > 
<asp:Content ID="Content2" ContentPlaceHolderID="ContentPlaceHolderl" 
runat="server"> 
<asp:Button ID="Buttonl"” runat="server" Text=" 修 改 图 书 基本 信息 " 
onc1lick="Button1l_ Click" /> 
<asp:Button ID="Button2" runat="server" Text=" 修 改 图 书 封面 图 片 " 
onclick="Button2 Click" /> 
<asp:Button ID="Button3" runat="server" Text=" 修 改 图 书 所 属 类 别 " 
onclick="Button3 Click" /> 
<asp:MultiView runat="server" ID="multiview" ActiveViewIndex="0" > 
<% 一 第 一 个 视图 ， 显 示 图 书 编辑 控件 --%> 
<asp:View runat="server" ID="view1"> 
<ucl:BookEditControl ID="controll" runat="server" /> 
</asp:View> 
<s-- 第 二 个 视图 ， 显 示 图 书 封面 编辑 控件 --s> 
<asp:View runat="server" ID="view2"> 
<uc2:BookCoverEdit id="control2" runat="server" /> 
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</asp:View> 
<$ 一 -第 三 个 视图 ， 显 示 图 书 类 别 编辑 控件 --%> 
<asp:View runat="server" ID="view3"> 
<uc3:BookCategoryControl id="control3" runat="server" /> 
</asp:View> 
</asp:MultiView> 
</asp:Content> 
//BookEdit .aspx.cs 
public partial class BookEdit : System.Web.UI.Page 
// 单 击 3 个 按钮 时 ， 分 别 切换 到 3 个 视图 
protected void Button1l Click (object sender, EventArgs e) 
| 


} 
protected void Button2 Click(object sender, EventArgs e) 


{ 


multiview.ActiveViewIndex = 0; 


multiview.ActiveViewIndex = 1; 


} 
protected void Button3 Click(object sender, EventArgs e) 


{ 


multiview.ActiveViewIndex = 2; 
} 
于 


图 书 编辑 页 面 BookEdit.aspx 的 运行 界面 ， 如 图 4.13 所 示 。 


上 
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图 413 图 书 编辑 页 面 


44 小 结 


本 章 综合 利用 前 面 几 章 所 介绍 的 知识 开发 了 一 个 网 上 书店 ， 主 要 功能 包括 图 书 浏览 、 
图 书 搜索 、 购 物 车 、 图 书 编辑 等 。 网 上 书店 是 一 种 典型 的 电子 商务 网 站 ， 熟 练 掌握 这 个 案 
例 对 于 巩固 ASP.NET 开发 技术 、 增 加 项 目 经 验 、 开 发 类 似 电子 商务 系统 都 有 很 大 帮助 。 
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软件 不 等 于 编码 ， 要 想 成 为 一 名 合格 的 软件 开发 人 员 ， 除 了 要 掌握 过 硬 的 编码 技术 以 
外 ， 还 需要 熟悉 规范 的 软件 开发 流程 ， 能 够 按照 软件 工程 思想 完成 软件 开发 全 过 程 。 本 章 
将 对 三 层 结构 、 源 码 管理 和 软件 测试 3 个 主题 进行 介绍 。 


5.1 源码 管理 简介 


在 软件 开发 过 程 中 ， 会 产生 大 量 的 源码 文件 。 这 些 源码 文件 分 散 于 整个 团队 中 ， 团 
队 中 很 多 成 员 都 可 能 会 不 定期 地 修改 源码 文件 中 的 任何 一 个 。 如 何 正确 地 管理 好 这 么 多 源 
码 文件 ， 协 调 各 个 开发 人 员 之 间 的 工作 ， 对 于 整个 软件 项 目 来 说 是 一 项 重要 工作 。 源 码 文 
件数 量 庞大 ， 修 改 频繁 ， 不 可 能 手工 维护 这 些 文件 ， 而 必须 依赖 于 源码 管理 软件 。 本 节 将 
以 与 Visual Studio 紧密 集成 的 微软 的 Visual SourceSafe 软件 为 例 ， 讲 解 源码 管理 各 个 任务 

源码 管理 主要 任务 包含 两 方面 : 一 是 跟踪 源码 修改 的 整个 过 程 , 保存 修改 的 历史 记录 ， 
并 可 以 获取 历史 记录 的 任 一 版 本 ， 可 以 对 任意 两 个 版 本 进行 比较 。 二 是 实现 多 个 开发 人 员 
协同 工作 ， 当 多 人 修改 同一 文件 时 保证 文件 的 一 臻 性， 避免 多 个 修改 之 间 相 互 覆 盖 。 

开发 人 员 在 整个 开发 过 程 中 都 可 能 对 源码 进行 修改 ， 而 新 的 修改 有 时 会 给 现 有 的 程 
序 引入 bug， 甚 至 有 时 所 做 修改 完全 是 方向 性 的 错误 ， 需 要 重新 确定 方案 重新 编码 。 当 这 
种 情况 发 生 时 ， 开 发 人 员 希 望 能 够 将 源码 恢复 成 以 前 未 修改 的 状态 ， 或 者 对 比 修改 前 后 的 
源码 有 何不 同 ， 从 而 较 容易 的 定位 bug 所 在 位 置 。 使 用 源码 管理 工具 可 以 方便 地 完成 这 些 
任务 。 

在 软件 开发 团队 中 ， 多 个 开发 人 员 共 同 完成 一 个 项 目 。 如 果 两 个 开发 人 员 张 三 和 李 
四 同时 修改 一 个 源 代码 文件 BookDAL.cs， 张 三 修改 了 其 中 的 一 个 方法 getBookByID， 而 
李 四 则 添加 了 另外 一 个 新 的 方法 deleteBook， 那 么 就 会 产生 两 个 不 同 版 本 的 BookDAL， 
每 个 版 本 只 包含 BookDAL 类 的 一 部 分 功能 ， 这 给 整个 软件 项 目的 管理 带 来 不 便 ， 并 很 可 
能 引发 其 他 错误 。 

为 了 避免 出 现 多 个 人 员 同 时 修改 一 个 文件 从 而 造成 版 本 混乱 的 情况 ， 应 该 对 文件 的 
修改 进行 控制 ， 可 以 采取 一 种 类 似 于 操作 系统 中 管理 临界 资源 的 方式 管理 源码 文件 ， 把 所 
有 源码 文件 集中 放置 在 源码 服务 器 上 ， 任 一 时 刻 最 多 只 能 有 一 名 开发 人 员 修 改 某 一 文件 。 

开发 人 员 张 三 在 修改 某 一 文件 BookDAL.cs 之 前 ， 要 进行 申请 ， 如 果 这 个 文件 当前 没 
有 人 修改 ， 则 批准 张 三 的 申请 ， 并 将 文件 BookDAL.cs 标记 为 正在 被 张 三 修 改 。 如 果 张 三 
申请 修改 BookDAL.cs 时 ， 这 个 文件 已 经 被 其 他 开发 人 员 申 请 修改 了 ， 则 拒绝 张 三 的 申 
请 ， 张 三 不 能 修改 此 文件 。 开 发 人 员 张 三 在 修改 完 BookDAL.cs 以 后 ， 要 释放 对 这 个 文件 
的 修改 权限 ， 并 重新 将 BookDAL.cs 标记 为 没有 人 修改 。 
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在 以 上 过 程 中 ， 开 发 人 员 申 请 并 修改 一 个 文件 的 过 程 称 为 “ 签 出 ”， 意 为 将 源码 文件 从 
服务 器 取出 来 给 开发 人 员 。 开 发 人 员 对 文件 修改 完成 以 后 ， 把 文件 交还 给 服务 器 并 标记 为 
当前 无 人 修改 的 状态 ， 这 个 过 程 称 为 “ 签 入 ”。 


5.2 使 用 Visual SourceSafe 管理 源码 


Visual SourceSafe( 简 称 VSS) 是 微软 推出 的 源码 管理 工具 , 当前 最 新 版 本 是 VSS 2005。 
微软 在 VSS 2005 以 后 又 推出 了 用 于 管理 团队 开发 的 软件 Team Foundation Server( 简 称 
TFS) ，TFS 除 包含 VSS 中 的 源码 管理 功能 之 外 ， 还 包括 其 他 更 多 团队 开发 的 相关 功能 。 
由 于 VSS 的 小 巧 (VSS 2005 安装 程序 只 有 105M) 和 TFS 的 庞大 (TEFS 2010 安装 程序 有 
2G 多 ) ，VSS 的 应 用 仍然 非常 普遍 。 


5.2.1 VSS 用 户 管理 


就 像 SQL Server 数据 库 一 样 , VSS 必须 有 合法 的 用 户 名 和 密码 才能 访问 。VSS 的 用 户 
有 两 种 不 同 的 权限 : 只 读 权限 和 读 写 权限 。VSS 安装 后 会 自动 添加 一 个 名 为 Admin 的 管理 
员 用 户 ， 拥 有 最 高 权限 。 使 用 Admin 登录 后 可 以 再 创建 和 修改 其 他 用 户 。 

VSS 2005 安装 后 ， 会 在 Windows 程序 菜单 中 生成 两 个 菜单 项 Microsoft Visual 
SourceSafe 和 Microsoft Visual SourceSafe Admin， 为 叙述 方便 ， 在 本 章 中 将 前 者 称 为 VSS 
程序 , 将 后 者 称 为 VSS 管理 程序 。 前 者 用 于 源码 管理 ,如 签 入 、 签 出 、 版 本 回 滚 和 比较 等 ， 
后 者 主要 用 于 VSS 配置 ， 如 添加 用 户 、 配 置 服 务 、 管 理 数据 库 等 。VSS 管理 程序 启动 后 主 
界面 如 图 5.1 所 示 。 

在 图 5.1 所 示 的 VSS 管理 程序 主 界面 中 ， 显 示 了 所 有 用 户 列表 及 状态 。 如 果 要 添加 用 
户 ， 则 从 菜单 中 选择 “用 户 ”|“ 添 加 用 户 ” 命 令 ， 则 弹出 如 图 5.2 所 示 的 “添加 用 户 ” 对 
话 框 。 在 其 中 输入 用 户 名 和 密码 ， 并 根据 情况 设置 是 否 只 读 ,然后 单 击 “ 确 定 ” 按 钮 即 可 。 


用 记名 中; | 
SoureeSafte ED [站 职 测 
厂 只 表 @ 各 助 中 


图 5.1 VSS 管理 程序 主 界面 图 5.2 添加 VSS 用 户 


如 果 要 删除 VSS 用 户 , 则 在 图 5.1 所 示 的 VSS 管理 程序 主 界面 中 选中 一 个 用 户 ， 从 菜 
单 中 选择 “用 户 ”|“ 删 除 用 户 ” 命 令 。 如 果 要 编辑 用 户 信息 ， 首 先 选 中 一 个 用 户 ， 再 从 
VSS 菜单 中 选择 “用 户 ”|“ 编 辑 用 户 ”命令 ， 会 弹出 “编辑 用 户 ” 对 话 框 ， 其 界面 与 图 
5.2 类 似 ， 在 其 中 设置 新 用 户 名 及 读 写 权 限 即 可 。 


5.2.2 管理 VSS 数据 库 


Visual SourceSafe 把 所 管理 的 文件 保存 到 特定 格式 的 VSS 数据 库 中 。 用 户 可 以 创建 、 
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删除 VSS 数据 库 ， 也 可 以 选择 当前 所 使 用 的 VSS 数据 库 。 
(1) 如 果 要 创建 VSS 数据 库 ， 从 VSS 管理 程序 中 选择 “文件 ”| “新 建 数据 库 ” 命 令 ， 
则 弹出 新 建 数据 库 向 导 对 话 框 ， 如 图 5.3 所 示 。 
(2) 单 击 “ 下 一 步 ” 按 钮 ， 然 后 选择 一 个 磁盘 路 径 作 为 VSS 数据 库 的 保存 路 径 ， 然 后 
一 直 单 击 “ 下 一 步 ” 按 钮 ， 直 到 出 现 如 图 5.4 所 示 的 选择 VSS 默认 设置 对 话 框 。 根据 VSS 
签 出 签 入 方式 的 不 同 ，VSS 有 “锁定 -修改 -解锁 ”模型 和 “复制 -修改 -合并 ”模型 两 种 不 同 
的 文件 修改 方式 : 
口 第 一 种 模式 下 每 个 文件 只 允许 一 个 用 户 签 出 ， 不 会 产生 多 个 用 户 同时 修改 一 个 文 
件 的 情况 ， 使 用 简单 ， 不 易 出 错 。 
口 第 二 种 模式 允许 多 个 用 户 同时 签 出 一 个 文件 并 进行 编辑 ， 然 后 将 多 个 用 户 的 修改 
进行 合并 。 这 种 模式 能 够 提高 多 用 户 并 发 性 。 


本 向 导 帮 | 一 个 从 本 机 访问 的 Visual 
Somressafe 测 指 库 。 


《< 上- 步 @) [下 一 步 加 站 取消 


图 5.3 新建 数据 库 向 导 图 5.4 选择 VSS 默认 设置 
(3) 如 果 VSS 拥有 多 个 数据 库 ， 可 以 通过 VSS 管理 程序 选择 当前 使 用 的 数据 库 。 具 
体操 作 步 又 为 从 菜单 中 选择 “文件 ”|“ 打 开 SourceSafe 数据 库 ” 命 令 ， 则 会 弹出 如 图 5.5 
所 示 对 话 框 ， 在 其 中 选择 想 要 打开 的 数据 库 ， 单 击 “ 打 开 ” 按 钮 即 可 。 
打开 SourceSafe 数据 库 x| 
可 用 的 数据 库 0): 打开 四 ) 
路 径 


G:AVSS_temp 


添加 必 ) 
移 除名 ) 
帮助 吕 


有 PS 〗《?《Z 
万 当 我 下 次 运行 衣 sual SourceSafe 时 打开 此 数据 库 


图 5.5 选择 VSS 数据 库 


(4) 为 了 保证 数据 安全 ， 还 可 以 备份 和 恢复 VSS 数据 库 。 要 备份 VSS 数据 库 ， 则 从 
VSS 管理 程序 中 选择 “存档 ”|“ 将 项 目 存档 ”命令 , 则 弹出 如 图 5.6 所 示 “ 选 择 要 存档 的 项 目 ” 
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对 话 框 。 在 其 中 选择 要 目录 或 者 根 目录 下 一 个 项 目 ， 单 击 “ 确 定 ”按钮 ， 则 进入 图 5.7 所 
示 的 “存档 向 导 ” 对 话 框 。 


Ed 
[3 存档 工具 会 将 这 些 项 目 放 到 
局 要 存档 的 项 目 @) 

人 @@): 自 | 四 synatarnput 奴 添加 人 ) 
时 wm | 

人 @) 
一 | 有 9 | 

图 5.6 选择 要 存档 的 项 目 图 5.7 存档 向 导 


(5) 在 图 5.7 所 示 的 存档 向 导 对 话 框 中 单 击 “ 下 一 步 ” 按 钮 ， 则 进行 如 图 5.8 所 示 的 
选择 存档 方式 对 话 框 。 有 3 个 选项 可 供 选择 ， 将 数据 库 备份 到 文件 、 备 份 并 从 数据 库 中 删 
除 、 仅 从 数据 中 删除 。 根 据 实 际 情况 选择 一 个 选项 ， 然 后 选择 备份 文件 的 保存 路 径 ， 单 击 
“下 一 步 ”按钮 ， 直 到 完成 为 止 。 

ED 到 


和 


区 数 括 保 仓 到 文件 太 】 
个 将 数据 保存 到 文件 , 然后 人 数据库 删除 以 节省 空间 到 ) 
个 永久 删除 文件 四) 


存档 文件 W) : 


图 5.8 选择 存档 方式 


(6) 当 VSS 数据 库 备 份 完成 后 ， 会 在 指定 的 路 径 下 生成 一 个 文件 ， 使 用 此 文件 可 以 恢 
复 VSS 数据 库 。 如 果 要 恢复 VSS 数据 库 ， 从 VSS 管理 程序 菜单 中 选择 “存档 ”|“ 恢 复 项 
目 ”命令 ， 则 弹出 恢复 向 导 ， 在 向 导 中 选择 要 恢复 的 文件 ， 一 直 单 击 “ 下 一 步 ”按钮 直到 
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5.2.3 配置 VSS 网 络 服务 


当 VSS 用 于 团队 开发 时 ， 团 队 成 员 需 要 从 局 域 网 基于 Intemet 上 访问 VSS 数据 库 ， 使 
用 VSS 所 提供 的 签 入 、 签 出 、 版 本 比较 等 功能 。 在 这 种 情况 下 ， 需 要 启用 VSS 的 网 络 服 
务 功能 。 操 作 步 骤 是 从 VSS 管理 程序 的 菜单 中 选择 “服务 器 ”|“ 配 置 ”命令 ， 则 弹出 如 
图 5.9 所 示 的 “服务 器 配置 ”对 话 框 。 


SourceSafe Internet | 局 域 网 | 


Vi sual SourceSafe Internet 允许 用 户 使 用 HTTF 连接 到 
Visual SourceSafes 


三 服务 器 (需要 管理 员 梭 限 ) 


伙 簿 用 这 各 计算 村 上 的 六 sual SourceSafe Internet (E) 
厂 需要 使 用 SSL 安全 连接 名) 


三 当前 数据 库 访问 
| 在 当前 数据 库 上 启用 SourceSafe Internet 四 ) 


Web 服务 器 名 称 (计算 机 名 或 IP 地 址 ) WH) 
Sunstudio 


注意 : 这 些 只 是 对 在 Vi sual Studi。 里 才 可 用 的 功能 的 设置 。 


图 5.9 VSS 服务 器 配置 对 话 杠 


如 果 要 在 Intemet 上 使 用 VSS， 则 在 SourceSafe Intemet 选项 卡 中 选中 相应 的 选项 ， 单 
击 “ 确 定 ” 按 钮 。 如 果 要 在 局 域 网 使 用 VSS， 则 在 “局 域 网 ”标签 页 中 进行 相应 的 设置 。 
设置 成 功 后 ， 即 可 通过 Internet 或 者 局 域 网 访问 此 服务 器 上 的 VSS 了 。 


和 提示 : 启用 网 络 服务 的 VSS 数据 库 必须 使 用 网 络 路 径 ( 如 \MyServer\VSSDB ) 而 不 能 
使 用 本 地 路 径 (如 D:\VSSDB )， 同 时 VSS 数据 库 所 在 目录 必须 设置 为 网 络 共享 
且 为 VSS 用 户 分 配 适 当 的 读 写 权限 。 


5.2.4 VSS 源码 管理 


VSS 以 项 目 为 单位 对 源码 进行 管理 ， 一 个 项 目 可 以 包含 若干 文件 夹 和 文件 。VSS 能 够 
跟踪 项 目 中 文件 的 签 入 签 出 状态 、 历 史 版 本 、 已 删除 文件 等 信息 。 

(1) 要 在 VSS 中 添加 一 个 项 目 ， 是 从 VSS 程序 菜单 中 选择 “文件 ”| “创建 项 目 ”命令 ， 
在 弹出 的 “创建 项 目 ”对 话 框 中 输入 项 目 名 称 ， 单 击 “确定 ”按钮 即 可 。 项 目 添 加 以 后 ， 通 
常 需要 为 其 设置 工作 目录 ，VSS 的 文件 签 出 时 就 存放 在 工作 目录 中 。 右 击 VSS 程序 中 一 个 项 
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目 ， 从 弹出 菜单 中 选择 “设置 工作 目录 ”命令 ， 即 可 为 VSS 项 目 指定 工作 目录 。 

(2) 在 VSS 中 创建 好 项 目 并 设置 好 工作 目录 以 后 , 接 下 来 要 做 的 是 把 需要 管理 的 文件 
添加 到 项 目 中 。 在 VSS 程序 中 ,选择 一 个 项 目 并 右 击 ， 从 弹出 的 快捷 菜单 中 选择 “添加 文 
件 ” 选 项 ， 弹 出 “添加 文件 ”对 话 框 。 在 其 中 选择 文件 并 单 击 “ 打 开 ” 按 钮 。 图 5.10 所 示 
为 添加 了 项 目 和 文件 以 后 的 VSS 程序 主 界面 。 


国 YVSs_tenp on Sunstudio - Yisual SourceSafe 六 览 器 
文件 四 编辑 至) 查看 WW) 版 本 处) 工具 CI) 站 点 他 帮助 0 
部 葬 马 X| 有 vv 藤 呈 加 中 | 站 -因子 弓 国 | 岛 固 名 


基 据 库 : VSS_tenp on Sunst 有 /test 的 内 容 厅 作 目录 : EVtenp\Datanput 
日 网 9/ 名 称 用 户 日 期 时 间 签 出 目录 
Besd Dipp.Confie 09-12-17 10:11 
Deass3DAL es 10-03-14 15:31 


LClass3Form Designer. cs 


signer. cs 
口 moaell ednx 
Drrogren. cs 09-12-17 10:09 


aaninistrator 
图 5.10 VSS 程序 主 界面 


(3) 在 图 5.10 所 示 的 VSS 程序 主 界面 中 , 列 出 了 test 项 目下 的 所 有 文件 ， 并 用 不 同 的 
图 标 表示 文件 的 签 入 、 签 出 状态 。 如 果 文件 被 签 出 ， 还 显示 签 出 文件 的 时 间 、 目 录 和 用 户 。 
要 想 签 出 并 编辑 文件 ， 在 VSS 程序 中 选中 要 编辑 的 文件 并 右 击 ， 则 弹出 如 图 5.11 所 示 的 
菜单 ， 从 菜单 中 选择 “编辑 ”选项 ， 则 弹出 如 图 5.12 所 示 的 编辑 对 话 框 。 


文件 $f/test/Class3DAL cs x 
兢 查看 WW.…. [你 想 
个 查看 该 文件 在 SourceSafe 上 的 副本 人) 取消 

苹 添加 文件 @&)..…. (ol 目 杂 并 编辑 这 个 六 伞 钙 ji -到 
园 出 六 六 件 列 表 和 外 ) F5 _ 和 助 中 | 
局 5 获取 最 新 版 本 @@) 使 用 
w 葵 出 @) 个 SourceSafe 编辑 器 (5) 
茸 符 入 加 会 注册 的 应 用 程序 BR); D:\. ..\ConmonTAIDEVdevenr exe 
有 描 销 签 出 四 eT | 
查看 历史 00)..… 
区 查看 差异 0 企 ys ， 这 对 您 的 计算 机 会 有 害处 。 安 全 
XX 册 除 CO SourceSafe . 
二 | 重 命名 四 二 
学 尾 竹 四， 厂 妈 当 按 下 shi ft 键 时 显示 该 对 话 框 目 ) 

图 5.11 VSS 文件 弹出 菜单 图 5.12 VSS 文件 编辑 对 话 框 


(4) 在 图 5.12 所 示 的 对 话 框 中 , 可 以 指定 要 打开 的 文件 所 使 用 的 编辑 器 , 单 击 “ 确 定 ” 
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按钮 后 ， 则 将 使 用 指定 的 编辑 器 打开 文件 。 在 编辑 器 中 修改 文件 结束 后 ， 从 VSS 程序 中 单 
文件 并 从 弹出 菜单 中 选择 “ 签 入 ”选项 ， 即 将 修改 后 的 文件 签 入 到 VSS 服务 器 。VSS 
中 的 文件 每 签 入 一 次 , 就 会 产生 一 个 新 的 版 本 。VSS 使 用 递增 的 整数 自动 为 各 个 版 本 命名 。 
在 图 5.11 所 示 的 文件 弹出 菜单 中 选择 “查看 历史 ”选项 , 则 弹出 如 图 5.13 所 示 的 文件 历史 
对 话 框 ， 在 其 中 显示 此 文件 的 所 有 历史 版 本 。 


[ln 


$/PublicPhone. root/PublicPhone/SJL Yeb/VserControls/EventIaEoC eae YT Se 


历史 外; 共 10 项 ) 


10-02-21 11:18 已 签 入 $/PublicPhone. root/PublicPhone/SJL Web/VserC 
10-02-21 10:48 已 等 入 $/PublicPhone root/PublicPhone/SIL Web/VserC 

09-12-27 22:44 已 签 入 $/PublicPhone. root/PublicFhone/SJL Web/UserC am© | 
09-12-01 21:32 已 签 入 $/PublicPhone. root/PublicPhone/SIL Web/Vsert [1 
09-10-29 12:18 已 签 入 3/Publicphone root/PublicFhone/SJL WebyUsert 

09-10-24 23:11 已 签 入 $/PublicPhone reot/PublicPhone/SJL Web/VUserC | 
09-10-24 22:12 已 签 入 $/PublicPhone. root/PublicPhone/SJL Web/UserC 条 住 性 ) 
09-10-24 18:17 已 答 入 $/PublicPhone. root/PublicPhone/SIL Web/VserC - 
09-10-24 10:40 已 外 建 mw | 


图 5.13 文件 历史 


(5) 在 图 5.13 的 文件 历史 对 话 框 中 右 侧 有 多 个 按钮 ， 可 以 完成 各 种 操作 。 选 中 一 个 版 
本 ， 单 击 “ 查 看 ”按钮 ， 可 以 显示 此 版 本 的 文件 内 容 。 单 击 “ 回 深 ” 按 钮 ， 则 可 以 将 文件 
恢复 到 指定 的 版 本 。 单 击 “ 获 取 ” 按 钮 ， 可 以 将 选中 版 本 导出 为 一 个 文件 。 在 文件 历史 中 
选中 两 个 不 同 版 本 ， 单 击 “ 差 异 ” 按 钮 ， 则 可 以 查看 两 个 版 本 文件 的 不 同 之 处 ， 如 图 5.14 
所 示 。VSS 用 不 同 的 颜色 显示 不 同 的 文件 差异 ， 红 色 表 示 被 删除 的 文本 ， 蓝 色 表 示 发 生 改 
变 的 文本 ， 绿 色 表 示 添 加 的 文本 ， 黑 色 表示 未 发 生变 化 的 文本 。 


ro] 版 本 5 和 zventInfoeContrel. ssez 版 本 6 的 相 导 夺 
TOIECITCICZID 


icPhone root/Publi cPhone/SJL Yeb/UserControls/Even.. [J $/FublicPhone. root/PublicFhone/SJL Yeb/UserContr 加 
1 CW Control Language="C#” AutoEventWireup="true” CodeBehind= 1 CW Control Language="C#" AutoEventWireup="true” CodeBehi 二 
Ee 2 Sa javascript”> 
3 Erte 
. ) SC 8theDute. ClientIDXY ) datepicker 0; 
scrip 
2 Ctable style="background:#aff; "> 了 <table style="background:#eff; "> 
3 《tr> ew 8 Ctr> 


9 Ctd colspan=0"> 
10 asp MiddenField II=~hidden0larD” ranat="server” /> 


Ca 
er 
onSumaary1” runat= -se 
1 Vt 
六 Lad Casp: Texthe Teri 13 sk dd Casp: Texthox Jp= 
12 5 Yd> Casp: TextBox runat="server” ID="eve | uv sp: Texthox runat="server” ID=" 
13 2 ID="theDate” runat="server 14 2 ID="theDate” runat="ser 
14 actd> 15 ta 
15 。。 《asp:DropDownList ID=“eventSourceList” ranat=“server”> (16 《asp:DropDownList ID="eventSoureeList” runat="server” 
16 Yasp:DropDornList> 17 asp:DropDovnList> 
17 YtdDQt3 委 对 信贷 tDCta> 18 Ytd>Ktd) 委 理 人 员 Ytd>Ctd> 
18 Casp'TextBox ID="nserlane” runat="server” Text="" ReadDn 19 <asp’DropDornList ID="userList” runat="server” Dataye 
a 2 YaropDomList> 
21 bs 
19 《tr>ta 浅 映 人 dtd>Ctd> 22 《tr>Xtd 小 用人 JtDXtD 
20 Lasp:Texthox ID="personliame" runat="server”>/asp:TextBo23 asp:TextBox ID="personliene” runat="server">C/asp:Tex 
21 Cdy<td) 电 话 GVtd><tay 24 《ftdyktd) 电 话 <yta>Gta> 
22 Lasp:TetBox TD="personPhne” runat="server”)C/asp:TextB25 。。 《asp:TextBox ID="personFhone” runat="server”)C/asp:Te 
23 本息 人 和 26 Ytd>Kt 由 地址 Ytd>Ctd> 


24 Casp:TextBox ID="personhddress” ranat="server"YC/asp:Tex27 ~ asp:TextBox ID="personAddress” ranat="server”>C/asp 
25 /td>Ktd) 满 意 程度 td>Ctd>Casp: DropDornList ranat="serve28 </td><t fd tdyCasp: DropDownList ronat="se 
26 《tr>Ktd) 尖 办 单位 Ytd><td> 29 《tr>Ktd) 尖 办 单位 Ytd><td>. 

27 asp:DropDownList ID="departaentList” ronat=“server” Dat 30 Casp:DropDownList ID="departaentList” ranat="server” 

28 Yes 如 orList>U4d》KtdD) 当 办 人 员 <td》Ctd>Casp-DropDo 31 。 /asp 了 DvnList>YtdDCtdD 汪 办 人 员 Ytd>Ctd>Casp:Droy 
| 
了 的 到 » 


图 5.14 VSS 显示 不 同 版 本 的 文件 差异 


=205% 


第 1 篇 ASPNET 网 络 开发 关键 技术 


5.2.5 集成 Visual Studio 与 Visual SourceSafe 


Visual Studio 开发 环境 和 Visual SourceSafe 都 是 微软 的 产品 ， 二 者 能 够 协调 地 配合 工 
作 。 在 Visual Studio 中 ， 集 成 了 Visual SourceSafe 的 客户 端 功能 ， 可 以 很 方便 地 实现 签 入 、 
签 出 、 查 看 历史 、 版 本 比较 等 功能 。 


全 提示 : 默认 情况 下 Visual Studio 2010 使 用 Team Foundation Server 实现 源码 管理 功能 。 
要 想 在 Visual Studio 2010 中 使 用 Visual SourceSafe 2005， 需 要 安装 一 个 插件 ， 下 
载 地 址 为 http://code.msdn.microsoft.com/KB976375。 


Visual Studio 2010 支 持 包括 TFS 和 VSS 在 内 的 多 个 源码 管理 软件 , 可 以 在 Visual Studio 
2010 中 进行 配置 ， 以 指定 一 种 源码 管理 工具 。 要 使 用 VSS 2005 作为 默认 源码 管理 工具 ， 
操作 步骤 为 从 菜单 中 选择 “工具 ”|“ 选 项” 命令， 在 打开 的 选项 对 话 框 中 ， 从 左 侧 的 树 型 
列表 中 选择 “ 源 代码 管理 ”|“ 插 件 选择 ”, 并 从 下 拉 表 框 中 选择 Microsoft Visual SourceSafe 

(如 果 要 连接 到 Internet 上 的 VSS, 则 选择 带 Intemet 的 Visual SourceSafe), 单 击 OK 按钮 ， 
如 图 5.15 所 示 。 


田 -环境 
由 -项 目 和 解决 方案 
日 - 源 代码 管理 


插件 设置 
由 文本 编辑 器 


四 
由 -IntelliTrace 


由 -文本 模板 化 


图 5.15 选择 源码 管理 插件 


如 果 要 把 Visual Studio 2010 中 的 项 目 〈 此 处 以 第 4 章 所 开发 的 BookShop 项 目 为 例 ) 
添加 到 VSS 源码 管理 ， 则 在 解决 方案 资源 管理 器 中 右 击 项 目 名 称 ， 从 弹出 的 快捷 菜单 中 选 
择 “ 将 解决 方案 添加 到 源 代 码 管理 ”选项 ， 则 会 弹出 VSS 登录 对 话 框 ， 如 图 5.16 所 示 。 

在 图 5.16 所 示 的 VSS 登录 对 话 框 中 ， 选 择 正确 的 VSS 数据 库 ， 输 入 合法 用 户 名 和 密 
码 ， 单 击 “ 确 定 ”按钮 ， 则 会 弹出 如 图 5.17 所 示 的 “添加 到 SourceSafe” 对 话 框 ， 让 用 户 
选择 项 目 在 VSS 中 的 存储 位 置 ， 一 般 使 用 默认 值 即 可 。 

在 图 5.17 所 示 对 话 框 中 选择 一 个 位 置 并 单 击 “ 确 定 ”按钮 后 ， 整 个 项 目 中 的 文件 就 被 
加 入 到 了 VSS 中 。 此 时 打开 VSS 程序 ， 就 可 以 看 到 这 个 项 目 ， 如 图 5.18 所 示 。 
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加 全 PublicFhone root 
回访 SJL root 


加 i 用 户 和 


用 PS QW: Waninistrator 
Ee 
| 


| 


oushep root 
| 
$/BoolShop. root 


取消 帮助 人 0 


图 5.16 VSS 登录 对 话 框 图 5.17 选择 项 目 在 VSS 中 的 存储 位 置 


文件 到 ) 编辑 时 查看 WW 版 本 G) 工具 人 站 点 @@ 帮助 0 
芒 苹 马 X| 训 vv 古 马 问 地 | 尹 - 因 于 弓 国 | 岛国 饭 
县 据 库 : VSS (7. /BookShop/BookShop 尚未 指定 工作 目录 
日 网 3 名 称 用 户 Bue E3 
由 上 AccountCheck roo 口 Aavancesearch aspx 10-03-07 10:55 < 
口 Aavaneeseareh aspx es 10-03-07 10:55 
ookShop 口 Aavancesearch aspx. des: 10-03-07 10:54 
9 BS WS DBestsell. aspx 10-03-05 21:02 


TSTeahine root 
田 和 访 Iis DBestsell. aspx. es 10-03-05 21:02 
ne DBestSell. aspx, desiener 10-03-05 21:02 
田 启 NRCN. root DBook cs 10-03-07 0:09 
向 rrc DBookDAL cs 10-03-07 9:44 
国 人 PublicPhone. root DBookDetail. aspx 10-03-07 0:16 
田 留 STL root 四 peupetsil emx.es 10-03-07 0:14 
DBookDetail. aspx. desimm, 10-03-06 11:47 
DBookListControl. asex 10-03-07 10:57 
MRookl i strontrol_ erny oe 1n-n3-n7_1n: 


加 司 
就 绪 


Taaninistrator 
图 5.18 VSS 中 新 添加 的 BookShop 项 目 


在 Visual Studio 中 将 项 目 添加 到 Visual SourceSafe 源码 管理 以 后 ， 在 解决 方案 资源 管 
理 器 中 文件 的 图 标 发 生 了 变化 ， 左 侧 添加 了 一 把 小 锁 〈- 昌 园 Default. aspx )， 表 示 此 文件 已 
经 被 签 入 VSS 服务 器 。 如 果 在 Visual Studio 中 编辑 这 种 被 签 入 的 文件 时 ， 文 件 会 自动 被 答 
出 ， 文 件 图 标 左 侧 的 小 锁 图 案 变 成 对 号 〈v 国 Default. aspx )， 表 示 此 文件 被 当前 开发 人 员 
签 出 。 如 果 文 件 被 其 他 开发 人 员 签 出 ， 则 文件 图 标 左 侧 会 出 现 一 个 小 人 头像 
(& 回 Default. aspx )。 

在 Visual Studio 中 进行 源码 文件 的 签 入 、 查 看 历史 、 文 件 比 较 、 获 取 版 本 等 操作 都 可 
以 通过 文件 右键 菜单 完成 ， 具 体操 作 方 式 与 单独 使 用 VSS 时 相同 。 


5.3 三 层 结 构 


随 着 软件 开发 技术 和 理论 的 发 展 ， 软 件 体系 架构 也 在 不 断 发 生 着 演化 ， 从 单机 程序 ， 
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到 客户 端 /服务 器 (C/S 结构 ) 程 序 , 到 浏览 器 /服务 器 (B/S 结构 ) 程 序 , 到 多 层 结构 (Multi-tier 
Architecture ) 程序 , 到 面向 服务 (SOA ) 程序 , 到 软件 即 服务 (SaaS ) 程序 , 到 云 计算 (Cloud 
Computing)。 软 件 体系 结构 的 发 展 经 历 了 多 个 的 阶段 ， 演 化 出 不 同 的 形式 ， 在 开发 一 个 具 
体 软件 项 目 时 ， 应 该 采用 何 种 体系 结构 ， 是 一 个 关系 全 局 的 重要 问题 。 本 节 将 介绍 当前 在 
Web 开发 中 广 为 应 用 的 三 层 结构 。 


5.3.1 三 层 结 构 概 述 


当 编 程 实现 一 个 功能 时 , 开发 人 员 最 直观 的 思路 是 把 所 有 代码 都 写 到 一 起 。 举例 来 说 ， 
如 果 用 ASPNET 实现 一 个 用 户 登录 页 面 时 ， 开 发 人 员 会 添加 一 个 ASPNET 页 面 ， 上 面 放 
置 相 应 的 控件 ， 然 后 在 页 面 的 后 台 代 码 中 编写 事件 处 理 程序 ， 将 用 户 输入 的 用 户 名 和 密码 
与 数据 库 中 的 数据 进行 验证 ， 根 据 正确 与 否 进行 适当 处 理 。 这 种 思路 虽然 简单 直观 ， 但 是 
仔细 分 析 却 存 在 很 大 的 缺陷 ， 功 能 混乱 、 不 能 复 用 、 难 以 测试 。 
软件 设计 和 开发 人 员 经 过 长 期 的 实践 、 思 索 和 研究 ， 提 出 了 软件 分 层 的 概念 ， 即 把 整 
个 软件 的 功能 分 成 多 个 层次 ， 并 形成 了 软件 的 多 层 架构 。 在 
软件 的 多 层 结构 中 ， 最 为 典型 的 是 三 层 结构 ， 即 把 软件 分 成 [天 
表现 层 、 业 务 罗 辑 层 、 数 据 访问 层 ， 各 层 之 间 通 过 业务 实体 | 
类 进行 数据 交换 。 典 型 的 三 层 结构 如 图 5.19 所 示 。 

在 图 5.19 所 示 的 三 层 结构 中 ， 表现 层 为 用 户 界面 , 对于。 lau 吉 昌 得 吕 怠 b2| 入 
ASPNET 程序 来 说 ， 表 现 层 就 是 页 面 。 业 务 罗 辑 层 处 理 业务 | Business 
规则 ， 例 如 ， 某 系统 规定 用 户 登录 时 连续 输 错 密码 不 得 超过 。 [DRCDAD) 

5 次 ， 否 则 用 户 账户 将 被 锁定 ， 这 就 是 一 个 业务 规则 。 数 据 。 Lee Assess Layer 
访问 层 负 责 与 数据 库 进行 数据 交换 ， 执 行 各 种 数据 库 命令 。 

图 5.19 中 的 箭头 表示 调用 和 依赖 关系 。 表 现 层 、 业 务 逻 (ean) 


辑 层 、 数 据 访问 层 各 层 之 间 是 单 向 的 逐 层 调用 关系 ， 不 能 跨 
层 调用 ， 也 不 能 反 向 调用 。 例 如 ， 表 现 层 只 能 调用 业务 罗 辑 

层 的 功能 ， 而 不 能 直接 调用 数据 访问 层 的 功能 ， 更 不 能 直接 图 5.19 三 层 结构 
操作 数据 库 。 数 据 访问 层 只 能 对 数据 库 进行 操作 ， 而 不 能 调 

用 业务 逻辑 层 的 功能 。 


5.3.2 ”银行 转账 实例 


为 了 更 加 形象 地 讲解 三 层 结构 的 设计 与 实现 , 本 节 将 引入 一 个 小 例子 : 银行 转账 程序 ， 
此 程序 可 以 将 资金 从 一 个 银行 账号 转换 到 另 一 个 银行 账号 。 为 了 突出 重点 ， 本 例 将 银行 转 
账 过 程 中 与 三 层 结构 无 关 的 功能 进行 了 省 略 ， 对 业务 规则 进行 了 加 强 。 经 过 如 此 修改 后 的 
银行 转账 程序 功能 描述 如 下 。 

银行 系统 中 每 个 客户 都 拥有 一 个 用 户 名 和 支付 密码 ， 一 个 客户 可 以 拥有 零 到 多 个 账 
号 。 各 个 银行 账号 之 间 可 以 相互 转账 。 转 账 时 资金 流出 的 账号 称 为 源 账号 ， 资 金 注 入 的 账 
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号 称 为 目的 账号 。 转 账 时 需要 输入 以 下 信息 : 源 账 号 、 源 账号 客户 名 、 源 账号 支付 密码 、 
转账 金额 、 目 的 账号 、 目 的 账号 客户 名 。 程 序 需 要 对 输入 的 信息 进行 以 下 验证 ， 如 果 全 部 
通过 才能 转账 成 功 。 

(1) 源 账 号 、 客 户 名 、 支 付 密码 都 正确 。 

(2) 源 账号 有 足够 的 资金 余额 。 

(3) 源 账 号 和 目的 账号 都 没有 被 冻结 。 

(4) 转账 金额 不 能 超过 某 一 上 限 ， 有 具体 数值 由 银行 指定 。 

(5) 目的 账号 用 户 名 和 密码 都 正确 。 

(6) 源 账号 当天 累计 转 出 资金 不 超过 某 一 上 限 ， 具 体 数 值 由 银行 指定 。 

(7) 源 账号 每 天 进行 转 出 交易 笔 数 不 超过 某 一 上 限 ， 具 体 数 值 由 银行 指定 。 

银行 系统 数据 库 包 括 以 下 4 个 表 。 

(1) 顾客 信息 表 Customer， 存 储 银行 顾客 信息 ， 表 中 各 字段 如 下 。 

口 CustomerID: 顾客 编号 ，nvarchar(20) 类 型 ， 主 键 。 

口 CustomerName: 顾客 名 称 ，nvarchar(10) 类 型 ， 不 可 为 空 。 

口 LoginPassword: 登录 密码 ，nvarchar(20) 类 型 ， 不 可 为 空 。 

口 PayPassword: 支付 密码 ，nvarchar(20) 类 型 ， 不 可 为 空 。 

(2) 银行 账户 表 BankAccount， 存 储 银行 账号 信息 ， 表 中 各 字段 如 下 。 

口 AccountNo: 银行 账号 ，nvarchar(20) 类 型 ， 主 键 。 

口 CustomerID: 顾客 编号 ，nvarchar(20) 类 型 ， 外 键 关 联 到 Customer 表 ， 不 可 为 空 。 

口 Balance: 账户 余额 ，float 类 型 ， 不 可 为 空 。 

口 IsLocked: 是 否 被 冻结 ，bit 类 型 ， 不 可 为 空 ， 默 认 值 为 false。 

(3) 账户 交易 表 AccountTransaction， 存 储 账号 交易 明细 ， 表 中 各 字段 如 下 。 

口 TransactionID: 交易 编号 ，int 类 型 ， 自 动 标识 列 ， 主 键 。 

口 TransactionDate: 交易 日 期 ，datetime 类 型 ， 默 认为 当前 时 间 ， 不 可 为 空 。 

口 SourceAccount: 源 账号 ，nvarchar(20) 类 型 ， 外 键 关 联 到 BankAccount 表 ， 不 可 

为 室 。 
口 DestinationAccount: 目的 账号 ，nvarchar(20) 类 型 ， 外 键 关联 到 BankAccount 表 ， 
不 可 为 空 。 

口 TransferMoney: 转账 金额 ，float 类 型 ， 不 可 为 空 。 

(4) 银行 政策 表 BankPolicy， 存 储 银 行政 策 〈 如 单 笔 转账 金额 上 限 、 每 日 转账 交易 上 
限 等 )， 表 中 各 字段 如 下 。 

口 PolicyID: 政策 ID，nvarchar(10) 类 型 ， 主 键 。 

口 _ Description: 政策 描述 ，nvarchar(50) 类 型 。 

口 PolicyContent: 政策 内 容 ， 如 交易 上 限 数值 ，nvarchar(50) 类 型 。 


5.3.3 ”未 分 层 的 银行 转账 程序 


通过 比较 多 层 体系 结构 与 不 分 层 的 软件 之 间 的 区 别 ， 能 够 更 好 地 体会 多 层 结 构 的 优 
势 。 本 节 先 采用 传统 的 不 分 层 的 方式 实现 上 述 银行 转账 程序 ， 并 分 析 这 种 方案 的 缺陷 ， 然 
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后 在 下 一 节 采 用 三 层 结构 对 银行 转账 程序 进行 重 构 。 

(1) 创建 一 个 ASPNET Web 应 用 程序 。 

(2) 在 SQL Server 中 添加 一 个 数据 库 ， 按 照 5.3.2 节 所 设计 的 数据 库 结构 在 数据 库 中 
添加 各 个 表 。 在 web.config 文件 中 添加 对 此 数据 库 的 连接 字符 串 。 

(3) 在 项 目 中 添加 一 个 页 面 SimpleBankPage.aspx， 在 页 面 上 放置 多 个 TextBox 供用 户 
输入 账号 、 密 码 、 金 额 等 数据 。 页 面 代 码 如 下 : 


<form id="forml" runat="server"> 

<div> 

<h3> 银 行 转账 程序 </h3> 

<table><tr> 

<td> 客 户 编号 : <asp:TextBox ID="customer" runat="server"></asp:TextBox></td> 
<td> 登 录 密码 : 

<asp:TextBox ID="loginPass" runat="server" TextMode="Password"> 
</asp:TextBox></td> 

<tq> 支 付 密码 : 

<asp:TextBox ID="payPass" runat="server" TextMode="Password"> 
</asp:TextBox></td> 

/EE><Er> 

<td> 转 出 账号 : <asp:TextBox ID="source" runat="server"></asp:TextBox></td> 
<td> 转 账 金额 : <asp:TextBox ID="money" runat="server"></asp:TextBox></td> 
<td></td></tr> 

tr 

<td> 接收 账号 : <asp:TextBox ID="destination" runat="server"> 
</asp:TextBox></td> 

<td> 账号 名 称 : <asp:TextBox ID="destinationName" runat="server"> 
</asp:TextBox></td> 

<td><asp:Button ID="ok" runat="server" Text=" 转 账 " Width="100" onclick= 
wok Click" /> 

</td></tr> 

</table> 

<asp:Label runat="server" ID="error" ForeColor="Red" Visible="false"> 
</asp:Label> 

<asp:Label runat="server" ID="message" Visible="false" ></asp:Label> 
</div> 

</form> 


(4) 在 SimpleBankPage.aspx.cs 中 编写 一 个 方法 ， 对 顾客 输入 的 客户 ID、 登 录 密 码 和 
支付 密码 进行 判断 ， 并 检测 该 用 户 是 否 拥 有 所 输入 的 源 账号 。 


/// <summary> 

/// 验证 顾客 信息 ， 客 户 ID、 登 录 密码 、 支 付 密码 、 是 否 拥有 源 账 号 
/// </summary> 

/// <returns> 验 证 合法 返回 true， 否 则 false</returns> 
private bool checkPassword() 


{ 


using (SqlHelper db = new SqlHelper()) 
I 
// 从 数据 库 读 取 顾 客 编号 和 两 个 密码 登录 密码 、 支 付 密码 ) 
string sql = "select LoginPassword,PayPassword from Customer where 
CustomerID=@id"; 
using (DbCommand command = db.GetSqlStringCommond(sql)) 


db.AddInParameter (command, "@id", DbType.String, customer. Text 


); 


using (DbDataReader reader = db.ExecuteReader (command)) 
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// 如 果 不 存在 此 顾客 ID， 则 显示 错误 提示 ， 返 回 false 
if (!reader.Read() ) 
| 
showError ("不 存在 你 所 输入 的 顾客 ID。") : 
return false; 
} 
string passl, pass2; 
passl = reader[0] .ToString(); 
pass2 = reader[1] .ToString(); 
// 比 较 两 个 密码 ， 如 果 不 正确 ， 则 显示 错误 提示 ， 返 回 false 
if (loginPass.Text != passl || payPass.Text != pass2) 
{ 
showError ("密码 不 正确 。"); 


return false; 


} 
} 
// 检 测 用 户 是 否 拥有 所 输入 的 源 账 号 
sql = "select count (x*) from BankAccount where AccountNo=@no and 
CustomerID=@id"; 
using (DbCommand command = db.GetSqlSstringCommond(sql)) 
{ 
db.AddInParameter (command, "@id", DbType.String, customer. Text 
); 
db.AddInParameter (command, "@no", DbType.String, source.Text); 
int n = Convert .ToInt32 (db.ExecuteScalar (Command) ); 
if (n == 0) 
{ 
showError (" 未 找到 你 所 输入 的 账号 。") ; 


return false; 


上 
} 
return true; // 所 有 检测 合法 ， 返 回 true 


(5) 在 SimpleBankPage.aspx.cs 中 添加 一 个 方法 ， 对 源 账 号 信息 进行 验证 ， 包 括 检验 
源 账号 是 否 被 冻结 ， 以 及 源 账 号 中 是 否 有 足够 的 余额 以 供 转账 。 
// 检 查 源 账号 是 否 被 冻结 以 及 资金 余额 是 否 充足 


private bool checkSourceAccount () 


{ 


using (SqlHelper db = new SqlHelper()) 
{ 
// 从 数据 库 获 取 账 户 状态 〈 是 否 冻结 ， 资 金 余额 ) 
string sql = "select IsLocked,Balance from BankAccount where 
AccountNo=@no"; 
using (DbCommand command = db.GetSqlStringCommond (sql) ) 
四 
db.AddInParameter (command, "eno"，DbType.String，source .Text) 
using (DbDataReader reader = db.ExecuteReader (command)) 
{ 
if (!reader.Read()) // 账 号 是 否 存 在 
| 
showError ("你 输入 的 账号 ["+source .Text+"] 不 存在 。") ; 
return false; 
} 
bool locked = Convert.ToBoolean (reader [0]); // 是 否 冻 结 
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double balance = Convert.ToDouble (reader[1]); // 账 号 余额 
reader.Close (); 
if (locked) 
i 
showError (" 源 账号 ["+source .Text+"] 已 经 被 冻结 。"); 
return false; 
| 
if (balance < double.Parse (money.Text) ) 


showError (" 源 账号 ["+source .Text+"] 资金 余额 不 足 。") ; 


return false; 


} 
} 


return true; 


(6) 在 SimpleBankPage.aspx.cs 中 添加 一 个 方法 ， 检 验 所 转账 金额 是 否 超出 数据 库 中 
定义 的 最 大 单 笔 转账 金额 ， 检 验 源 账 号 当天 转账 次 数 是 否 已 经 达到 最 大 转账 次 数 ， 检 验 此 
次 转账 是 否 会 使 源 账 号 的 转账 总 额 超出 最 大 当日 转账 金额 。 


/// <summary> 


/// 检查 交易 上 限 (包括 单 笔 金额 上 限 ， 累 计 转 账 金额 上 限 ， 累 计 交 易 次 数 上 限 ) 


/// </summary> 


/// <returns> 检 查 是 否 合法 </returns> 
private bool checkTransactoinLimit () 


{ 
using (SqlHelper db = new SqlHelper()) 
{ 
double singleTranserLimit; // 单 笔 转 账 金额 上 限 
int transferTimesLimit; // 单 日 累计 转账 次 数 上 限 
double transferMoneyLimit; // 单 日 累计 转账 金额 上 限 


// 从 数据 库 读 取 银 行政 策 (3 个 上 限 值 》 

string sql = "select PolicyContent from BankPolicy where PolicyID 
=@id"; 

using (DbCommand command = db.GetSqlstringCommond(sql)) 

{ 


// 从 数据 库 读 取 单 笔 转账 金额 上 限 

db.AddInParameter (command, "@id", DbType.Sstring, "policy001"); 
singleTranserLimit = Convert.ToDouble (db.ExecuteScalar 
(command)); 

// 从 数据 库 读 取 单 日 累计 转账 次 数 上 限 

command.Parameters["@id"] .Value = "policy002"; 
transferTimesLimit = Convert.ToInt32 (db.ExecuteScalar 
(command)); 

// 从 数据 库 读 取 单 日 累计 转账 金额 上 限 

command. Parameters ["Qid"] .Value = "policy003"; 
transferMoneyLimit = Convert .ToDouble (db.ExecuteScalar 
(command) ) > 


1 

// 检 验 单 笔 转账 上 限 

if (singleTranserLimit < double.Parse (money.Text)) 

i 
showError ("你 要 转账 的 金额 超过 单 笔 转 账 上 限 "+singleTranserLimit. 
ToString ()); 
return false; 
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// 从 数据 库 读 取 源 账号 当日 累计 转账 次 数 和 当时 累计 转账 金额 
sql = "select count (*) , sum(TransferMoney) from AccountTransaction " 
+" where SourceAccount=eno" 7 
using (DbCommand command = qdb.GetSqlStringCommond(sql) ) 
db.AddInParameter (command, "eno"，DbType.String，source -Text) 
using (DbDataReader reader = db.ExecuteReader (command)) 
{ 
if (reader.Read()) 
{ 
int times = Convert.ToInt32 (reader [0]); 
double theMoney=0; 
if(reader[1]!=DBNul1.Value) 
theMoney= Convert .ToDouble (reader[1]); 
if (times == transferTimesLimit) // 检 查 累计 转账 次 数 
{ 
showError ("账号 ["+source.Text 
+"] 当日 累计 转账 次 数 已 经 达到 上 限 ， 不 能 进行 转账 。") ; 


return false; 


} 
// 检 查 累 计 转 账 金额 
if (theMoney+double.Parse (money.Text) > transferMoneyLim- 
it) 
{ 
showError ("此 次 转账 将 使 账号 [" 
+ source.Text + "] 当 日 累计 转账 金额 超出 上 限 ， 不 能 进行 转 
账 。") ; 
return false; 
} 
BE 
} //using DbDataReader 
} //using DbCommand 


//using SqlHelper 


return true; 


(7) 在 SimpleBankPage.aspx.cs 中 添加 一 个 方法 ， 检 测 目的 账号 是 否 合法 ， 包 括 目的 
rn 


账号 是 否 存在 ， 账 号 所 对 应 的 客户 名 称 与 输入 的 是 否 一 致 ， 目 的 账号 是 否 


1 
已 经 被 冻结 。 


/// <summary> 

/// 检查 目的 账号 〈 账 号 是 否 存在 ， 名 称 是 否 正确 ， 是 否 被 冻结 ) 
/// </summary> 

/// <returns> 检 查 是 否 合 法 </returns> 

private bool checkDestinationAccount () 


{ 


using (SqlHelper db = new SqlHelper()) 


{ 


// 从 数据 库 获取 账户 状态 〈 是 否 冻结 ， 顾 客 ID， 顾客 名 称 ) 
String sql = "select IsLocked, CustomerID from BankAccount where 
AccountNo=@no"; 
bool locked; 
string customerid, customerName; 
using (DbCommand command = db.GetSqlstringCommond(sql)) 
a 
db.AddInParameter (command, "@no", DbType.String, destination. 
Text); 
using (DbDataReader reader = db.ExecuteReader (command)) 
{ 
if (!reader.Read()) 
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} 


sql = 


showError ("你 输入 的 账号 ["+destination.Text+"] 不 存在 。 
return false; 

上 

locked = Convert.ToBoolean (reader [0]); 

customerid = Convert.ToString (reader [1]); 


"select CustomerName from Customer where CustomerID=@id"; 


using (DbCommand command = db.GetSqlStringCommond (sql) ) 


是 


1£ 


!; 


db.AddInParameter (command, "@id", DbType.String, customerid); 
customerName = Convert.ToString (db.ExecuteScalar (Command) ); 


(customerName != destinationName.Text) // 输 入 目的 账号 名 称 不 对 
showError (" 目 的 账号 ["” + source -Text + "] 名 称 不 对 ， 请 检查 账号 是 否 正 确 。 
中 避 

return false; 


(locked) // 账 号 被 冻结 


showError ("目的 账号 [" + source.Text + "] 已 经 被 冻结 。") ; 
return false; 


return true; 


(8) 在 SimpleBankPage.aspx.cs 中 添加 一 个 方法 ， 完 成 具体 的 转账 操作 ， 


内 容 : 


减少 源 账 
个 操作 应 该 在 同 


号 金额 、 增 加 目的 账号 金额 、 忆 录 交易 详情 。 为 了 保 va ， 
-事务 中 完成 。 


// 执 行 转账 操作 


private void transerMoney () 


{ 


.214 . 


using (SqlHelper db = new SqlHelper()) 


{ 


using (Trans tran = new Trans()) // 启 动 事务 


1 


// 修 改 源 账号 余额 

string sql = "update BankAccount set Balance=Balance-@money where 

RccountNo=eno"7 

using (DbCommand command = db.GetSqlStringCommond (sql)) 
db.AddInParameter (command, "@no", DbType.String,destination. 
Text); 
db.AddInParameter (command, "@money", DbType.Double, money. 
Text); 
db.ExecuteNonQuery (command, tran); 

有 

// 修 改 目 的 账号 余额 

sql = "update BankAccount set Balance=Balance+Qmoney where 

AccountNo=@no™; 

using (DbCommand command = db.GetSqlstringCommond (sql) ) 

{ 
db.AddInParameter (command, "@no", DbType.Sstring, source. 
Text); 
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db.AddInParameter (command, "@money", DbType.Double, money. 
Text); 
db.ExecuteNonQuery (command, tran); 
} 
// 将 交易 记录 保存 到 数据 库 
sql = "insert into AccountTransaction (SourceAccount, Destination 
Account, TransferMoney) "+ " values (@source,@dest,@money)"; 
using (DbCommand command = db.GetSqlSstringCommond (sql)) 
{ 
db.AddInParameter (command, "@source", DbType.SsString, source. 
Text); 
db.AddInParameter (command, "@dest", DbType.String, destina— 
tion.Text); 
db.AddInParameter (command, "@money", DbType.Double, money. 
Text); 
db.ExecuteNonQuery (command, tran); 
} 
tran.Commit (); // 提 交 事务 


上 
(9) 在 “转账 ”按钮 的 Click 事件 中 ， 调 用 以 上 方法 ， 实 现 所 有 验证 和 转账 操作 。 


protected void ok Click(object sender, EventArgs e) 
{ 
if (!checkPassword()) 
return; 
if (!checkSourceRAccount () ) 
return; 
if (!checkTransactoinLimit()) 
return; 
if (!checkDestinationAccount()) 
return; 
transerMoney (); 
showMessage ("转账 操作 成 功 。") ; 


(10) 在 SimpleBankPage.aspx.cs 中 添加 两 个 方法 ， 分 别 用 于 显示 错误 信息 和 交易 成 功 
的 提示 信息 。 


/// <summary> 
/// 显示 错误 提示 ， 同 时 隐藏 操作 结果 提示 
/// </summary> 
/// <param name="errorMessage"> 要 显示 的 错误 信息 </param> 
private void showError (String errorMessage) 
{ 
error.Text = errorMessage; 
error.Visible = true; 
message.Visible = false; 
message.Text = ""; 
} 
/// <summary> 
/// 显示 操作 提示 (如 操作 成 功 ) ， 同 时 隐藏 错误 提示 
/// </summary> 
/// <param name="content"> 要 显示 的 操作 提示 内 容 </param> 
private void showMessage (string content) 


message.Visible = true; 
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error.Visible = false; 
error.Text = ""» 
message.Text = content; 


上 
(11) 运行 程序 ， 运 行 界面 如 图 5.20 所 示 。 
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银行 转账 程序 

客户 编号 ，[110101198001011234 ”登录 密码 ,[ 支付 密码 ,|[ 
转 出 账号 ，[TestAccount002 转账 金额 : [50 

接收 账号 ，[Testhccount002 账号 名 称 , FS 转账 
未 找到 你 所 输入 的 账号 。 
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图 5.20 银行 转账 程序 运行 界面 


5.3.4 未 分 层 程序 的 缺陷 


5.3.3 节 的 未 分 层 的 银行 转账 程序 虽然 实现 了 所 要 求 的 功能 ， 但 是 存在 以 下 缺陷 。 

(1) 功能 耦合 。 在 银行 转账 页 面 中 ， 要 处 理 顾 客 信 息 、 账 号 信息 、 转 账 交易 、 银 行政 
策 、 用 户 界面 等 各 方面 的 功能 ， 不 符合 面向 对 象 设计 原则 中 的 单一 职责 原则 ， 是 一 种 不 好 
的 设计 。 

(2) 不 可 复 用 。 如 果 做 一 个 完整 的 网 上 银行 系统 ， 则 在 银行 转账 页 面 中 包含 的 一 些 功 
能 可 能 在 其 他 多 个 页 面 中 用 到 。 例 如 ， 验 证 顾客 密码 功能 在 登录 页 面 中 会 用 到 ， 查 询 账 户 
余额 功能 在 账户 详情 页 面 中 会 用 到 ， 查 询 账户 是 否 锁 定 在 账户 的 其 他 交易 中 会 用 到 。 由 于 
这 些 代 码 与 银行 转账 页 面 紧密 耦合 在 一 起 ， 就 无 法 在 其 他 页 面 中 调用 ， 而 是 需要 将 这 些 代 
码 复制 到 其 他 页 面 中 ， 并 且 进 行 适当 修改 。 这 种 做 法 违反 了 一 个 功能 只 实现 一 次 的 软件 设 
计 原 则 ， 如 果 要 对 某 一 功能 做 修改 ， 则 需要 修改 相似 代码 的 多 个 复 本 ， 既 增加 了 工作 量 ， 
而 且 容 易 遗 漏 。 

(3) 不 易 修改 。 如 果 由 于 业务 规则 发 生变 化 或 者 数据 库 结构 发 生变 化 而 需要 修改 代码 ， 
由 于 代码 分 散在 几乎 所 有 的 页 面 中 ， 很 难 找到 并 改正 所 有 需要 修改 的 地 方 。 

(4) 文件 庞大 。 由 于 功能 复杂 ， 银 行 转账 页 面 代码 数量 太 多 ， 不 易 阅 读 和 维护 。 

(5) 弱 类 型 。 由 于 使 用 DataReader (或 者 DataSet) 操作 数据 ， 得 到 的 数据 总 是 object 
类 型 ， 在 具体 使 用 时 需要 进行 频繁 的 类 型 转换 。 这 样 一 方面 代码 编写 太 复杂 ， 另 一 方面 当 
项 目 规模 较 大 时 ， 开 发 人 员 很 难 记得 住 每 个 字段 的 类 型 ， 容 易 引起 类 型 转换 错误 。 


5.3.5 三 层 结构 的 银行 转账 程序 


正如 前 面 所 分 析 ， 当 使 用 早期 的 不 分 层 的 软件 结构 开发 较为 复杂 的 系统 时 ， 暴 露出 种 
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种 次 端 。 使 用 分 层 的 软件 体系 结构 来 开发 软件 ， 能 够 克服 这 些 浆 端 ， 设 计 出 优秀 的 软件 。 
在 软件 的 多 层 结构 中 ， 最 为 经 典 的 是 三 层 结 构 ， 包 括 表 现 层 、 业 务 逻 辑 层 和 数据 访问 层 。 

当 用 .NET 开发 应 用 程序 时 ， 通 常 每 一 层 是 一 个 单独 的 项 目 ， 其 中 表现 层 是 ASPNET 
项 目 (或 者 Windows 应 用 程序 项 目 )， 业 务 逻辑 层 和 数据 访问 层 都 是 类 库 项 目 。 另 外 ， 还 
需要 一 个 单独 的 类 库 项 目 , 封装 所 有 的 业务 实体 类 。 用 ASPNET 创建 多 层 应 用 程序 的 第 一 
个 步骤 就 是 在 一 个 解决 方案 中 创建 以 上 4 个 项 目 。 以 实现 银行 转账 程序 为 例 ， 创 建 三 层 结 
构 的 项 目 步骤 如 下 。 

(1) 创建 一 个 ASPNET Web 应 用 程序 BankApp。 

(2) 在 同一 解决 方案 中 添加 一 个 类 库 项 目 BankBLL， 作 为 业务 逻辑 层 。 

(3) 在 同一 解决 方案 中 添加 一 个 类 库 项 目 BankDAL， 作 为 数据 访问 层 。 

(4) 在 同一 解决 方案 中 添加 一 个 类 库 项 目 BankEntity， 作 为 业务 实体 类 库 。 

(5) 在 以 上 4 个 项 目 之 间 建 立 引 用 和 依赖 关系 ，BankBLL 引用 BankDAL，BankApp 
引用 BankBLL， 三 个 项 目 都 引用 BankEntity。4 个 项 目 之 间 的 引用 关系 如 图 5.21 所 示 。 


BankApp 一 引用 -| BankBLL 一 引用 一 一 BankDAL 


引用 


引用 BankEntity 引用 


图 5.21 银行 转账 程序 各 项 目 之 间 引用 关系 


搭建 好 4 个 项 目 以 后 ， 接 下 来 的 工作 就 是 将 整个 程序 功能 分 配 到 各 个 层 。 表 现 层 负 责 
处 理 用 户 界面 (如 显示 信息 、 接 收 用 户 输入 )， 业 务 逻 辑 层 实现 业务 逻辑 (如 本 例 中 的 银行 
业务 规则 )， 数 据 访 问 层 负责 操 作 数 据 库 〈 执 行 增删 改 查 )， 业 务实 体 层 包 含 项 目 中 的 业务 
实体 类 。 在 较为 简单 的 情况 下 ， 数 据 库 中 的 一 个 表 对 应 于 一 个 实体 类 ， 表 中 每 个 字段 对 应 
实体 类 的 一 个 属性 。 

(6) 在 BankEntity 中 添加 4 个 实体 类 ， 代 码 如 下 。 


全 提示 : 在 Visual Studio 中 添加 类 时 ， 上 默认 情况 下 class 关键 字 前 面 没有 访问 修饰 符 ， 此 
时 类 的 访问 级 别 为 intemal， 只 能 在 当前 程序 集中 访问 。 对 于 实体 类 来 说 ， 需 要 
在 其 他 项 目 访问 ， 所 以 在 class 关键 字 前 面 添加 public 关键 字 ， 将 类 声明 为 公 书 
共 的 。 


// 银 行 客户 实体 类 
public class Customer 
{ 
public string id { get; set; } 
public string name { get; set; } 
public string loginPassword { get; set; } 
public string payPassword { get; set; } 


本 
// 银 行 账 号 实体 类 
public class BankAccount 


* 1 


第 1 篇 ASPNET 网 络 开发 关键 技术 


public string no { get; set; } 
public double balance { get; set; } 
public string customerID { get; set; } 
public bool isLocked { get; set; } 

} 

// 账 号 交易 信息 实体 类 

public class AccountTransaction 

{ 
public int transactionID { get; set; } 
public string sourceAccount { get; set; } 
public string destinationAccount { get; set; } 
public DateTime transactionDate { get; set; } 
public double transerMoney { get; set; } 

// 银 行政 策 实体 类 

public class BankPolicy 
public string id { get; set; } 
public string description { get; set; } 
public string content { get; set; } 

. 


(7) 在 数据 访问 层 BankDAL 项 目 中 添加 一 个 CustomerDal 类 ,完成 与 顾客 信息 相关 的 
数据 访问 。 本 例 中 ， 对 于 顾客 信息 的 数据 访问 只 涉及 一 个 功能 ， 即 根据 顾客 编号 得 到 顾客 
详情 ， 因 此 CustomerDal 类 中 只 有 一 个 方法 。 


public static class CustomerDal 
{ 
// 根 据 顾客 ID 得 到 顾客 信息 
public static Customer getById(string id) 
{ 
Customer customer = null; 
using (SqlHelper db = new SqlHelper()) 
{ 
// 构 建 SQL 语句 
string sql = "select CustomerName, LoginPassword, PayPassword from 
Customer where CustomerID=@id"; 
using (DbCommand command = db.GetSqlSstringCommond (sql)) 
{ 
db.AddInParameter (command, "@id", DbType.Sstring, id); 
using (DbDataReader reader = db.ExecuteReader (command)) 
下 
// 将 从 数据 库 读 取 的 数据 转换 成 Customer 类 的 实例 
if (!reader.Read()) 
return null; 
customer = new Customer () 7 
customer.id = id; 
customer.name = reader["CustomerName"] .ToString(); 
customer.loginPassword = reader["LoginPassword"]. 
ToString(); 
customer .payPassword = reader["PayPassword"] .ToString(); 


} 
} 


return customer; 


+ 
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(8) 在 数据 访问 层 BankDAL 项 目 中 添加 一 个 BankAccountDal 类 ， 完 成 银行 账户 相关 
的 数据 访问 。 本 例 中 用 到 两 个 功能 ， 一 个 功能 是 根据 银行 账号 得 到 账号 详情 ， 另 外 一 个 功 
能 是 修改 账户 余额 。 
public static class BankAccountDal 
// 根 据 账号 得 到 账号 详细 信息 
public static BankAccount getById (string no) 
{ 


1 


null; 
new SqlHelper()) 


BankAccount account 
using (SqlHelper db 


1 


{ 
// 构 建 查询 用 的 select 语句 
string sql = "select CustomerID,Balance, IsLocked from BankAccount 
where AccountNo=@no"; 
using (DbCommand command = db.GetSqlSstringCommond (sql)) 
a 
db.AddInParameter (command, "@no", DbType.String, no); 
using (DbDataReader reader = db.ExecuteReader (command)) 
让 
// 将 从 数据 库 读 取 的 数据 转换 成 BankAccount 类 的 实例 
if (!reader.Read()) 
return null; 
account = new BankAccount (); 
account.no = no; 
account .customerID = reader["CustomerID"] .ToString(); 
account .balance = Convert.ToDouble!( reader["Balance"]); 
account .isLocked = Convert.ToBoolean (reader["IsLocked"]); 
lL 
} 


return account; 


/// <summary> 
/// 向 账户 中 增加 金额 
/// </summary> 
/// <param name="account"> 要 增加 金额 的 账户 </param> 
/// <param name="money"> 增 加 的 数额 (负数 则 减少 ) </param> 
public static void addMoney (string account, double money) 
{ 
using (SqlHelper db = new SqlHelper()) 
{ 


string sql = "update BankAccount set Balance=Balance+emoney where 
AccountNo=@no"; 
using (DbCommand command = db.GetSqlSstringCommond (sql)) 


{ 
db.AddInParameter (command, "@no", DbType.string, account); 
db.AddInParameter (command, "@money", DbType.Double, money); 
db.ExecuteNonQuery (command); 


;3 


(9) 在 数据 访问 层 BankDAL 项 目 中 添加 一 个 PolicyDal 类 ， 完 成 对 银行 政策 的 数据 访 
问 功能 。 
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public static class PolicyDal 


和 


-Es 


#region 常用 政策 ID 
private const string ConstSingleTransferLimitID = "policy001"; 


// 单 笔 转 账 上 限 ID 


private const string ConstTranserTimesLimitID = "policy002"; 
// 单 日 交易 次 数 上 限 ID 

private const string ConstTranserMoneyLimitID = "policy003"; 
// 单 日 转账 总 额 上 限 ID 

#endregion 


/// <summary> 

/// 根据 政策 ID 得 到 政策 内 容 

/// </summary> 

/// <param name="id"> 政 策 ID</param> 

/// <returns> 政 策 内 容 </returns> 

public static BankPolicy getById(string id) 


{ 
BankPolicy policy = null; 
using (SqlHelper db = new SqlHelper()) 
{ 


string sql = "select Descroption,PolicyContent from BankPolicy 


where PolicyID=@id"; 
using (DbCommand command = db.GetSqlStringCommond (sql)) 
{ 
db.AddInParameter (command, "@id", DbType.Sstring, id); 
using (DbDataReader reader = db.ExecuteReader (command)) 
i 
if (!reader.Read()) 
return null; 
// 将 数据 库 中 读 取 的 数据 转换 成 实体 类 BankPolicy 
policy = new BankPolicy(); 
policy.id = id; 


policy.description = reader["Description"].ToString(); 


policy.content = reader["PolicyContent"] .ToString(); 


} 
} 
return policy; 
} 
// 得 到 单 笔 转账 金额 上 限 
public static double getSingleTransferLimit () 
{ 
BankPolicy policy = getById (ConstSingleTransferLimitID); 
return double.Parse (policy.content); 
} 
// 得 到 单 日 转账 交易 次 数 上 限 
public static int getDailyTransferTimesLimit () 
{ 
BankPolicy policy = getById (ConstTranserTimesLimitID) 
return int.Parse(policy.content); 
} 
// 得 到 单 日 累计 转账 金额 上 限 
public static double getDailyTransferMoneyLimit () 
{ 
BankPolicy policy = getById (ConstTranserMoneyLimit1ID); 
return double.Parse (policy.content); 
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(10) 在 数据 访问 层 BankDAL 项 目 中 添加 一 个 TransactionDal 类 ， 完 成 对 转账 交易 明 
细 的 数据 访问 功能 。 


public static class TransactionDal 
{ 
// 根 据 交易 ID 得 到 交易 详情 
public static AccountTransaction getBYID (int id) 
AccountTransaction tran = null; 
using (SqlHelper db = new SqlHelper()) 
下 


// 构 建 查询 用 的 select 语句 
string sql = "select SourceAccount, DestinationAccount, 
TransactionDate, TransferMoney from AccountTransaction where 
TransactionID=@id"; 
using (DbCommand command=db.GetSqlStringCommond(sql)) 
{ 
db.AddInParameter (command, "@id", DbType.Int32, id); 
using (DbDataReader reader=db.ExecuteReader (command)) 
| 
if (!reader.Read()) // 未 找到 此 ID 
return null; 
// 根 据 读 取 的 数据 构建 一 个 实体 类 
tran = new AccountTransaction(); 
tran.transactionID = id; 
tran.sourceAccount = Convert.ToString (reader 
["SourceAccount"]); 
tran.destinationAccount = Convert.ToString (reader 
["DestinationAccount"]); 
tran.transactionDate = Convert.ToDateTime (reader 
["TransactionDate"]); 
tran.transerMoney = Convert.ToDouble (reader 
["TransferMoney"]); 


3 
h 
return tran; 
| 
/// <summary> 
/// 添加 新 的 交易 信息 
/// </summary> 
/// <param name="tran"> 交 易 信 息 </param> 
/// <returns> 添 加 的 行 数 </returns> 
public static int addTransaction (AccountTransaction tran) 
i 
using (SqlHelper db=new SqlHelper()) 
{ 
// 构 建 插入 SQL 语句 
string sql = "insert into AccountTransaction " 
+"(SourceAccount, DestinationAccount,TransactionDate, 
TransferMoney) " 
+" values (@source, @dest, @date, @money) "; 
// 创 建 一 个 命令 ， 并 添加 各 个 参数 
using (DbCommand command = db.GetSqlSstringCommond (sql)) 
{ 
db.AddInParameter (command, "@source", DbType.String, tran. 
sourceAccount); 
db.AddInParameter (command, "@dest", DbType.Sstring, tran. 
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“Ee 


} 
WA 


destinationAccount); 

db.AddInParameter (command, "@date", DbType.DateTime, tran. 
transactionDate); 

db.AddInParameter (command, "Q@money", DbType.Double, tran. 
transerMoney); 


return db.ExecuteNonQuery (command); // 返 回 受 影响 的 行 数 


<summary> 


/// 根据 账号 和 交易 时 间 获 得 交易 详情 列表 


Va 
LA 


</summary> 
<param name="account"> 银 行 账号 </param> 


/// <param name="from"> 查 询 的 起 始 日 期 时 间 </param> 

/// <param name="to"> 查 询 的 终止 日 期 时 间 </param> 

/// <returns> 符 合 条 件 的 交易 列表 </returns> 

public static List<AccountTransaction> getByAccountAndDate (string 
account, DateTime from, DateTime to) 


List<AccountTransaction> list = new List<RccountTransaction>() 7 
using (SqlHelper db = new SqlHelper()) 


{ 


} 


// 构 建 查询 数据 的 select 语句 


string sql = "select TransactionID, DestinationAccount, 
TransactionDate, TransferMoney " 


+ "from AccountTransaction where (SourceAccount=@account) " 
+" and (TransactionDate between @datel and @date2) "; 


using (DbCommand command = db.GetSqlstringCommond (sql)) 


{ 


// 以 下 三 条 语句 向 查询 语句 添加 参数 
db.AddInParameter (command, "@account", DbType.Sstring, 
account); 
db.AddInParameter (command, "@datel", DbType.DateTime, from); 
db.AddInParameter (command, "@date2", DbType.DateTime, to); 
using (DbDataReader reader = db.ExecuteReader (command)) 
i 
// 针 对 查询 到 的 数据 循环 
// 每 条 数据 生成 一 个 AccountTransaction 类 ， 添 加 到 列表 中 
while (reader.Read()) 
| 
Var tran = new AccountTransaction(); 
tran.transactionID = Convert.ToInt32 (reader 
["TransactionID"]); 
tran.sourceAccount = account; 
tran.destinationAccount = Convert.ToString( reader 
["DestinationAccount"]); 
tran.transactionDate = Convert.ToDateTime (reader 
["TransactionDate"]); 
tran.transerMoney = Convert.ToDouble (reader 
["TransferMoney"]); 
list.Add (tran); 


return list; 
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(11) 在 业务 逻辑 层 BankBLL 项 目 中 添加 一 个 CustomerBll 类 ， 实 现 银 行 顾客 相关 的 
业务 逻辑 。 由 于 本 例 中 没有 与 顾客 相关 的 业务 逻辑 ， 所 以 CustomerBll 类 仅 是 调用 数据 访 
问 层 CustomerDal 的 相关 功能 。 


public static class CustomerB11 


有 


public static Customer getById(string id) 
{ 


return CustomerDal.getById(id); 
} 


a 


(12) 在 业务 多 辑 层 BankBLL 项 目 中 添加 一 个 BankAccountBll 类 ， 实 现 银行 账户 相关 
的 业务 逻辑 。 本 例 中 与 银行 账户 相关 的 一 条 业务 规则 有 一 条 ， 当 银行 账户 被 冻结 后 ， 则 不 
能 修改 其 账户 余额 。 

public static class BankAccountBl11 


{ 


public static BankAccount getById (string no) 
{ 


return BankRccountDal.getById (no) 
} 


public static void addMoney (string account, double money) 


{ 
BankAccount theAccount = BankAccountB]l1]l .getById(account); 
if (theAccount.isLocked) 


throw new BankException ("账户 ["+account+"] 被 冻结 , 不 能 执行 此 操作 。") ; 
BankAccountDal .addMoney (account, money); 


) 


以 上 语句 中 用 到 的 一 个 异常 类 BankException 为 自 定义 异常 类 ， 代 码 如 下 : 
// 银 行业 务 异 常 类 

[Serializable] 

public class BankException : ApplicationException 


{ 
public BankException() { } 
public BankException( string message ) : base( message ) { } 
public BankException( string message, Exception inner ) : base (message, inner ) 
而 |， 
protected BankException( 
System.Runtime.Serialization.SerializationInfo info, 


System.Runtime .Serialization.StreamingContext context ) : base( info, 
Context ) { } 


1 

(13) 在 业务 逻辑 层 BankBLL 项 目 中 添加 一 个 PolicyBll 类 ,实现 银行 政策 相关 的 业务 
逻辑 。 由 于 本 例 没 有 有 具体 与 银行 政策 相关 的 业务 规则 ，PolicyBll 类 的 作用 仅 是 调用 数据 访 
问 层 相 应 的 方法 。 


public static class PolicyB11 


public static double getSingleTransferLimit () 
{ 


return PolicyDal.getSingleTransferLimit () 7 


} 
public static int getDailyTransferTimesLimit() 
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return PolicyDal.getDailyTransferTimesLimit(); 


} 
public static double getDailyTransferMoneyLimit () 


return PolicyDal.getDailyTransferMoneyLimit (); 
} 


(14) 在 业务 逻辑 层 BankBLL 项 目 中 添加 一 个 TransactionBl 类 ,实现 银行 转账 相关 的 
业务 四 辑 。 本 例 中 ， 银 行 转账 有 复杂 的 业务 逻辑 ， 如 单 笔 转账 金额 封顶、 当日 累计 转账 金 
额 封顶 、 当 日 累计 转账 次 数 最 大 值 等 。 在 TransactionBll 类 中 要 实现 所 有 这 些 业 务 规则 。 


public static class TransactionBll 


{ 
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public static AccountTransaction getBYID (int id) 
return TransactionDal.getByID(id) 
} 
public static List<AccountTransaction> getByAccountAndDate (string 
account, DateTime from, DateTime to) 
L 
return TransactionDal .getByAccountAndDate (account, from, to); 
} 
/// <summary> 
/// 执行 银行 转账 业务 
/// </summary> 
/// <param name="customerId"> 顾 客 ID</param> 
/// <param name="from"> 源 账户 </param> 
/// <param name="to"> 目 的 账户 </param> 
/// <param name="toName"> 目 的 账户 的 顾客 名 称 </param> 
/// <param name="money"> 转 账 金额 </param> 
public static void transferMoney (string customerId, string from, string 
to, string toName,double money) 
{ 
BankAccount source = BankAccountB]1] .getById (from); 
// 得 到 源 账 户 信息 
if (source.customerID != customerId) 
throw new BankException ("顾客 不 拥有 所 请 求 账户 。"); 
if (source.balance < money) 
throw new BankException ("账户 余额 不 足 "); 
var list = TransactionBl] .getByAccountAndDate (from, DateTime.Today, 
DateTime.Today .AddDays (1) .AddSeconds (-1)); 
int n = PolicyBll .getDailyTransferTimesLimit (); 
if (list.Count>=n) 
throw new BankException ("账户 今天 交易 次 数 已 经 达到 上 限 "); 
if (money<=0) 
throw new BankException ("转账 金额 必须 大 于 零 "); 
double d = PolicyB11.getSingleTransferLimit (); 
if (money>=d) 
throw new BankException ("转账 金额 超出 单 笔 转账 上 限 ") ; 
d = list.Sum(tr => tr.transerMoney); 
// 计 算 今天 转账 总 额 
if (dtmoney>PolicyB11.getDailyTransferMoneyLimit () ) 
throw new BankException (" 此 次 转账 将 使 账户 今天 转账 总 金额 将 超过 上 限 ， 转 
账 失败 ") ; 
BankAccount dest = BankAccountB11.getById (to) 
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// 得 到 目的 账户 信息 
string name = CustomerB11.getById (dest.customerID) .name; 
if (name!=toName) 
throw new BankException ("目的 账户 名 称 不 对 。") ; 
// 将 所 有 数据 修改 操作 增加 余额 、 减 少 余额 、 记 录 交 易 明 细 ) 封装 在 一 个 事务 中 
using (System.Transactions.TransactionScope scope = new System. 
Transactions.TransactionSscope()) 
{ 
BankRAccountB11.addMoney (from，-money) ; // 减 少 源 账 户 余额 
BankRAccountB11.addMoney (to, money); // 增 加 目的 账户 余额 
AccountTransaction tran = new AccountTransaction(); 
tran.sourceAccount = from; 
tran.destinationAccount = to; 
tran.transerMoney = money7 
tran.transactionDate = DateTime.Now; 
TransactionDal.addTransaction (tran); // 记 录 交 易 信息 
scope.Complete(); // 成 功 结束 事务 


(15) 在 表现 层 ASPNET Web 应 用 程序 BankApp 中 , 添加 一 个 页 面 TierBankPage.aspx， 


在 页 面 上 放置 相应 的 控件 供用 户 输入 数据 。 


<h3> 银 行 转账 程序 〈 三 层 结构 版 ) </h3> 

<table><tr> 

<td> 客 户 编号 : <asp:TextBox ID="customer" runat="server"></asp:TextBox></td> 
<td> 登 录 密码 : <asp:TextBox ID="loginPass" runat="server" TextMode="Password"> 
</asp:TextBox></td> 

<td> 支 付 密码 : <asp:TextBox ID="payPass" runat="server" TextMode="Password"> 
</asp:TextBox></td> 

/Er><Er> 

<td> 转 出 账号 : <asp:TextBox ID="source" runat="server"></asp:TextBox></td> 
<td> 转 账 金额 : <asp:TextBox ID="money" runat="server"></asp:TextBox></td> 
<td></td></tr> 

<tr> 

<td> 接收 账号 ; <asp:TextBox ID="destination" runat="server"></asp:TextBox> 
</td> 

<td> 账号 名 称 : <asp:TextBox ID="destinationName" runat="server"></asp:TextBox> 
</td> 

<td><asp:Button ID="ok" runat="server" Text=" 转 账 " Width="100" onclick= 
Sor erick® /> 

</td></tr> 

</table> 

<asp:Label runat="server" ID="error" ForeColor="Red" Visible="false"> 
</asp:Label> 

<asp:Label runat="server" ID="message" Visible="false" ></asp:Label> 


(16) 在 TierBankPage.aspx 页 面 中 ”转账 ”按钮 的 Click 事件 中 , 调用 业务 逻辑 层 相应 


功能 实现 银行 转账 ， 如 果 发 生 错误 ， 则 显示 错误 提示 。 


protected void ok Click(object sender, EventArgs e) 


Var C = CustomerBl]l .getById(customer.Text); // 得 到 顾客 信息 
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TE Te == 
i 
showError ("用 户 不 存在 。"); 


return; 


if (c.loginPassword != loginPass.Text || c.payPassword != payPass.Text) 
showError ("密码 不 正确 。"); 
try 
上 
// 调 用 业务 罗 辑 层 类 实现 转账 功能 
BankBLL .TransactionB11.transferMoney (Customer .Text, source .Text， 
destination.Text, destinationName.Text, double.Parse (money. 
Text)); 
showMessage ("转账 操作 成 功 。"); 
} 
catch (BankEntity.BankException ex) // 发 生 自 定义 异常 
{ 


} 
catch (Exception ex) // 发 生 其 他 异常 
t 


showError (ex.Message); 


showError ("发 生 未 知 错误 。"+ex.Message); 
} 
private void showError (string errorMessage) // 显 示 错误 信息 


error.Text = errorMessage; 
error.Visible = true; 
message.Visible = false; 
message.Text = " "7 
} 
private void showMessage (string content) 
// 显 示 交 易 成 功 提示 
{ 
message.Visible = true; 
error.Visible = false; 
error.Text = ""; 
message.Text = content; 


5.3.6 ”三 层 结 构 程序 的 优势 


对 比 未 分 层 的 银行 转账 程序 和 三 层 结构 的 银行 转账 程序 ， 三 层 结构 有 如 下 优势 。 

(1) 功能 内 聚 。 在 三 层 结构 的 银行 转账 与 数据 库 相 关 的 代码 集中 在 数据 访问 层 ， 而 且 
涉及 具体 每 个 表 的 操作 都 集中 在 一 个 类 中 ， 其 他 各 层 也 具有 相同 特点 。 这 种 设计 使 得 类 和 
项 目的 功能 具有 很 高 的 内 聚 性 ， 符 合 页 面 对 象 设计 原则 。 

(2) 代码 复 用 。 由 于 数据 访问 和 业务 逻辑 都 封装 特定 的 类 和 方法 中 ， 存 储 于 类 库 中 ， 
无 论 在 任何 地 方 ( 如 不 同 的 页 面 、 甚 至 不 同 的 项 目 中 〉 需 要 这 些 功 能 ， 都 可 以 添加 对 类 库 
的 引用 ， 调 用 相应 方法 即 可 ， 遵 守 了 一 个 功能 只 实现 一 次 的 原则 。 

(3) 不 易 修改 。 由 于 相似 功能 在 一 个 类 中 集中 存放 ， 而 不 是 分 散 于 整个 项 目 众多 的 页 
面 后 台 代 码 中 ， 使 得 代码 修改 更 加 简单 。 例 如， 假设 数据 库 中 银行 顾客 表 的 某 个 字段 名 
称 发 生 了 改变 ,那么 只 需要 修改 数据 访问 层 的 CustomerDal 类 中 的 相关 代码 即 可 ,数据 访 
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问 层 其 他 类 、 业 务 罗 辑 层 、 表 现 层 代码 都 不 用 作 任 何 改动 。 

(4) 文件 规模 适中 。 由 于 将 整个 程序 功能 分 成 三 层 ， 使 得 每 一 部 分 的 功能 相对 简单 ， 
限制 了 文件 规模 ， 不 会 出 现 不 易 阅 读 和 维护 的 庞大 源码 文件 。 

(5) 强 类 型 。 由 于 业务 实体 类 封装 数据 库 中 数据 ， 用 类 的 属性 表示 数据 表 中 的 字段 ， 
从 而 实现 了 强 类 型 ， 保 证 了 类 型 安全 ， 还 可 以 充分 利用 Visual Studio 的 智能 提示 功能 ， 提 
高 编程 效率 。 例 如 ， 当 使 用 DataTable 封装 BankAccount 表 中 的 数据 时 ， 如 果 要 访问 账户 
余额 ， 则 需要 使 用 以 下 语句 : 


double balance=Convert .ToDouble (talbe.Rows [0] ["Balance"]); 


上 述 语 句 中 ， 需 要 记 住 数据 表 中 的 字段 名 称 和 类 型 ， 还 需要 进行 类 型 转换 。 如 果 字 段 
名 称 记 错 或 者 打字 时 出 现 拼写 错误 ， 由 于 字段 名 在 字符 串 中 ，Visual Studio 不 会 进行 语法 
检测 ， 因 此 识别 不 出 错误 的 字段 名 。 

在 三 层 结构 中 ， 用 实体 类 BankAccount 封装 了 BankAccount 表 中 数据 ， 同 样 如 果 要 访 
问 账 户 余 额 ， 则 需要 以 下 代码 。 

BankAccount account=BankAccountDal.getByID(" 账 号 "); 

double balance=account .balance; 

上 述 代码 BankAccount 类 及 balance 属性 都 是 强 类 型 , 在 Visual Studio 开发 环境 中 , 能 
够 自动 提示 balance 属性 并 进行 类 型 检测 ， 提 高 了 编程 效率 ， 避 免 了 类 型 错误 。 


5.4 单元 测试 


单元 测试 又 称 为 模块 测试 ， 是 针对 程序 模块 (如 方法 ) 来 进行 正确 性 检验 的 测试 工作 。 
单元 测试 主要 是 用 来 检验 程序 的 内 部 逻辑 ， 通 常 由 撰写 代码 的 编写 者 负责 进行 。 当 开发 人 
员 编 写 完 成 程序 中 的 一 个 功能 (在 .NET 中 此 功能 通常 对 应 于 一 个 方法 ) 后 ,应 该 写 一 段 测 
试 代码 以 验证 其 正确 性 ， 这 个 工作 就 是 单元 测试 。 


5.4.1 单元 测试 概述 


单元 测试 是 一 项 很 重要 的 工作 ， 有 助 于 尽早 发 现 程序 中 的 bug。 根 据 软件 工程 理论 ， 
软件 中 bug 发 现 的 时 间 越 晚 ， 修 改 这 个 bug 所 花费 的 代价 就 越 高 ， 所 以 应 该 尽早 发 现 程序 
中 的 错误 并 及 时 修改 。 用 于 单元 测试 的 代码 也 是 整个 软件 的 一 个 重要 组 成 部 分 ， 需 要 妥善 
保存 并 在 后 续 开 发 过 程 中 使 用 。 实 践 证 明 ， 对 于 程序 某 个 地 方 的 修改 可 能 会 在 无 意 中 影 响 
到 其 他 地 方 从 而 导致 程序 出 错 。 因 此 ， 在 对 软件 进行 了 较 大 的 改动 以 后 ， 不 但 应 该 测试 被 
修改 的 代码 ， 还 要 对 与 之 相关 的 所 有 代码 都 重新 执行 一 遍 测试 ， 以 确保 程序 的 修改 不 会 引 
入 新 的 bug。 

单元 测试 是 软件 开发 过 程 中 一 个 必 不 可 少 的 环节 ， 对 于 保证 软件 质量 有 着 重要 作用 ， 
这 一 点 在 测试 驱动 开发 (Test-driven development) 中 体现 较为 明显 。 测 试 驱动 开发 是 一 种 
现代 的 软件 开发 方法 ， 利 用 测试 来 驱动 软件 程序 的 设计 和 实现 。 测 试 驱动 开发 是 极限 编程 
中 倡导 的 程序 开发 方法 , 其 指导 思想 简单 来 说 就 是 在 编写 实现 一 个 功能 以 前 先 写 测试 程序 ， 
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然后 再 编码 使 其 通过 测试 。 


软件 开发 有 两 个 基本 的 衡量 指标 : 实现 的 功能 和 质量 。 测 试 驱动 开发 就 像 两 顶 帽子 思 


考 法 的 开发 方式 ， 先 戴 上 实现 功能 的 帽子 ， 在 测试 的 辅助 下 ， 快 速 实现 ] 


E 确 的 功能 ， 再 戴 


上 重 构 的 帽子 ， 在 测试 的 保护 下 ， 通 过 去 除 元 余 和 重复 的 代码 ， 提 高 代码 重用 性 ， 实 现 对 
质量 的 改进 。 可 见 测试 在 测试 驱动 开发 中 确实 属于 核心 地 位 ， 贯 穿 了 开发 的 始终 。 


5.4.2 ”创建 和 运行 单元 测试 


在 Visual Studio 中 可 以 完成 包括 单元 测试 在 内 的 各 种 测试 。 测 试 也 是 一 种 代码 ， 通 过 
执行 这 些 测试 代码 调用 被 测试 的 模块 ， 并 根据 执行 结果 判断 测试 是 否 通过 。 进 行 测试 时 经 


常用 到 以 下 几 个 类 。 


口 TestClassAttribute 类 : 这 是 一 个 属性 〈Attribute) 类， 所 有 的 测试 类 都 必须 添加 这 


个 属性 。 


口 TestMethodAttribute 类 : 这 也 是 一 个 属性 (Attribute) 类 ， 所 有 的 需要 独立 执行 的 


测试 方法 都 必须 添加 这 个 属性 。 


口 TestContext 类 : 表示 测试 上 下 文 ， 用 于 存储 提供 给 单元 测试 的 信息 。 
口 Assert 类 : 该 类 包含 计算 布尔 值 条 件 的 一 组 静态 方法 ， 如 AreEqual、 IsTrue、IsNull 
等 。 如 果 此 条 件 计算 为 tue， 则 断言 通过 。 如 果 所 验证 的 条 件 不 为 tue， 则 上 断言 将 


失败 ， 测 试 不 通过 。 


下 面 以 5.3 节 所 创建 的 三 层 结构 的 银行 转账 程序 为 例 ， 说 明 创建 和 运行 单元 测试 的 步 
又 。 由 于 单元 测试 的 方法 基本 相同 ， 此 处 只 选择 数据 访问 层 中 几 个 比较 典型 的 类 和 方法 进 


(1) 在 Visual Studio 中 打开 银行 转账 程序 解决 方案 ， 打 开 BankDAL 项 目 中 的 
CustomerDal 类 ， 在 类 名 上 右 击 ， 从 弹出 的 快捷 菜单 中 选择 ”创建 单元 测试 ”选项 ， 如 图 


5.22 所 示 。 
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图 5.22 创建 单元 测试 


(2) 之 后 会 弹出 如 图 5.23 所 示 的 “创建 单元 测试 ”对 话 框 ， 对 话 框 中 自动 选中 了 
CustomerDal 类 中 所 有 方法 ，Visual Studio 将 会 为 选中 的 方法 自动 生成 单元 测试 代码 框架 。 
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图 5.23 “创建 单元 测试 ”对 话 框 


(3) 单 击 “确定 ”按钮 ， 则 Visual Studio 会 自动 生成 一 个 CustomerDalTest 类 ， 类 中 包 
含 主要 代码 如 下 。 


外 提 示 : 如 果 是 第 一 次 在 解决 方案 中 添加 测试 ，Visual Studio 会 添加 一 个 新 的 测试 项 目 ， 
并 将 新 生成 的 测试 类 添加 到 测试 项 目 中 。 如 果 不 是 第 一 次 添加 测试 ， 则 会 把 测试 
类 添加 到 已 有 的 测试 项 目 中 。 


[TestClass()] // 说 明 这 是 一 个 测试 类 
public class CustomerDalTest 
| 
private TestContext testContextInstance; // 测 试 上 下 文 
public TestContext TestContext 
{ 
get 
{ 


return testContextInstance; 


Set 
{ 
testContextInstance = value; 

} 
} 
[TestMethod ()] // 说 明 这 是 一 个 测试 方法 
public void getByIdTest () // 测 试 方法 框架 
. 

string id = string.Empty; // TODO: 初始 化 为 合适 的 值 

Customer expected = null; // TODO: 初始 化 为 合适 的 值 

Customer actual; 

actual = CustomerDal .getById (id); 

Assert .AreEqual (expected, actual); 

Assert.Inconclusive ("Verify the correctness of this test method."); 
} 
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(4) 修改 自动 生成 的 测试 方法 getByIdTest 为 以 下 内 容 。 


[TestMethod () ] 

public void getByIdTest () 
string id = "110101198001011234"; 
Customer actual; 
actual = CustomerDal .getById(id); // 调 用 被 测试 方法 
Assert.IsNotNull (actual); // 断 言 返回 结果 非 空 
// 以 下 代码 将 Customer 对 象 的 各 个 字段 与 数据 库 中 数据 进行 比较 ， 以 验证 其 正确 性 
Assert.AreEqual (id, actual.id); 
Assert .AreEqual ("路 人 甲 "，actual .name); 
Assert.AreEqual ("123456", actual.loginPassword); 
Assert.AreEqual ("12345600", actual.payPassword); 


(5) 将 BankApp 项 目 中 的 配置 文件 和 App_Data 目录 下 的 数据 库 文件 复制 到 测试 项 目 
BankDalTest 中 。 


全 提示 : 测试 运行 时 ， 会 从 测试 项 目 而 非 被 测试 项 目 中 寻找 配置 文件 。 因此， 如果 项 目 中 
用 到 配置 文件 ， 一 定 要 记 住 把 同样 的 配置 文件 复制 到 测试 项 目 中 。 
如 果 项 目 使 用 Sql Server 2005 Express 版 数据 库 , 并 且 连 接 字 符 串 使 用 了 相对 路 径 
引用 数据 库 (如 AttachDbFileName=[DataDirectory]lbankmdf )， 则 测试 时 需要 将 
数据 库 文件 复制 到 测试 项 目 中 。 


(6) 右 击 getByIdTest 方法 内 部 ， 从 弹出 的 快捷 菜单 中 选择 “运行 测试 ”选项 ， 如 图 
5.24 所 示 。 
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图 5.24 运 和 


(7) 运行 测试 后 ， 会 在 Visual Studio 开发 环境 底部 “测试 结果 ”视图 中 显示 测试 运行 
结果 ， 显 示 通 过 与 否 。 如 图 5.25 所 示 为 getByIdTest 测试 通过 时 的 情况 。 


图 5.25 ”测试 通过 
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(8) 为 了 演示 测试 不 通过 的 情况 ， 修 改 CustomerDal.getCustomerById 中 的 代码 以 产生 
错误 ， 例 如 将 以 下 SQL 语句 中 的 CustomerID 字段 名 故意 写 错 为 ID。 


select CustomerName,LoginPassword,PayPassword from ” Customer where 
CustomerID=@id 


再 次 运行 测试 ， 则 测试 结果 如 图 5.26 所 示 。 


EE haninistratoresUiSTUDIO 2010-! -| 99 运行“ 岂 油 车 |9y”% 名 | 


@。 测 趟 二 行 未 通过 结果 ; 0/1 个 通过 ; 选中 的 项 : 1 


图 5.26 测试 失败 


在 测试 结果 中 双击 未 通过 的 测试 ， 则 会 显示 测试 详情 及 未 通过 原因 ， 如 图 5.27 所 示 。 


图 5.27 测试 详情 


从 图 5.27 中 可 以 看 到 测试 未 通过 的 原因 是 运行 过 程 中 产生 了 SqlException 异常 “ 列 名 
ID 无 效 ”， 并 且 能 够 看 到 异常 发 生 的 代码 所 在 行 ， 从 而 能 够 较 容易 找到 并 改正 错误 。 

(9) 要 测试 一 个 方法 ， 应 该 测试 其 在 各 种 情况 下 的 执行 情况 ， 例 如 ， 当 接受 正常 参数 
和 非法 参数 时 。 对 getCustomerById0 方 法 也 要 用 各 种 情况 进行 测试 。 在 测试 类 
CustomerDalTest 中 再 添加 一 个 方法 ， 代 码 如 下 。 

[TestMethod] 


public void getByIdTest2 () 
{ 


string id = "Not Exists"; // 数 据 库 中 不 存在 此 ID 
Customer actual; // 断 言 返回 结果 为 空 
actual = CustomerDal .getById (id); // 调 用 被 测试 方法 


Assert.IsNull (actual); 


(10) 运行 新 的 测试 并 使 其 通过 ， 如 果 有 错误 就 修改 代码 。 
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5.4.3 ”管理 单元 测试 


在 真实 的 软件 项 目 中 ， 会 产生 大 量 测试 类 和 测试 方法 ， 应 该 采用 有 效 的 组 织 方式 管理 
这 些 测 试 。Visual Studio 中 的 测试 列表 是 一 种 很 方便 的 管理 测试 的 方法 。 有 些 测 试 之 间 存 
在 某 种 关系 ， 例 如 同属 于 对 同一 个 数据 访问 类 的 测试 ， 或 者 都 是 与 某 一 功能 相关 的 测试 ， 
可 以 按照 这 些 关 系 将 一 组 测试 组 合 在 一 起 ， 形 成 测试 列表 。 下 面 演 示 了 创建 和 使 用 测试 列 
表 的 具体 步 又 。 

(1) 从 Visual Studio 菜单 中 选择 “测试 ”|“ 创 建新 的 测试 列表 ”命令 , 则 弹出 如 图 5.28 
所 示 的 对 话 框 ， 在 其 中 输入 测试 列表 名 称 ， 并 选择 测试 列表 位 置 。 

(2) 单 击 OK 按钮 后 ， 会 打开 如 图 5.29 所 示 的 “测试 列表 编辑 器 ”窗口 。 
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图 5.28 创建 新 的 测试 列表 图 3.29 测试 列表 编辑 器 


(3) 在 “测试 列表 编辑 器 ”窗口 中 ， 从 左 侧 选择 “列表 中 未 列 出 的 测试 ”， 则 右 侧 会 
列 出 不 在 当前 测试 列表 中 的 测试 ， 可 以 将 测试 拖 动 到 左 侧 的 一 个 测试 列表 中 。 本 例 将 
CustomerDalTest 类 中 的 两 个 测试 getByIdTest 和 getByIdTest2 拖 动 到 MyTest 列表 中 ， 如 图 
5.30 所 示 。 


和 J 册 | 分 地 要 孝 : -J [所 有 列 】 -| 人 入 关键 于 > -jE 


图 530 将 测试 加 入 测试 列表 


(4) 从 “测试 列表 编辑 器 ”窗口 右 侧 选 中 需要 运行 的 测试 ， 并 单 击 工具 栏 上 的 运行 测 
试 按钮 串 即 可 运行 所 选中 的 测试 。 
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5.4.4 ”代码 覆盖 率 


为 了 提高 测试 可 靠 性 ， 测 试 需要 执行 到 目标 方法 中 的 所 有 代码 。 例 如 ， 如 果 目 标 方法 
中 有 记 …else… 语 句 ， 则 测试 时 让 和 else 分 支 中 的 代码 都 应 该 被 测试 到 。Visual Studio 中 经 
过 配置 以 后 可 以 显示 测试 覆盖 率 ， 具 体操 作 步 又 如 下 。 

(1) 在 解决 方案 资源 管理 器 中 ， 打 开 Solution Items 文件 来， 双击 Local.testsettings 项 
目 ， 如 图 5.31 所 示 。 

(2) 之 后 会 打开 如 图 5.32 所 示 的 “测试 设置 ”对 话 框 。 在 对 话 框 的 左 侧 列表 中 选择 “ 数 
据 和 诊断 ”选项 ， 从 右 侧 选中 “代码 覆盖 率 ” 复 选 枉 ， 如 图 5.32 所 示 ， 然 后 单 击 “应 用 ” 
按钮 。 


EEE 2 


岛 二 
同 W 吕 方案 Develop” 6 个 项 目 ) 
Ttems 


olution 


| 


图 5.31 ”local.testsettings 所 在 位 置 图 5.32 测试 设置 
(3) 在 图 5.32 所 示 的 “测试 设置 ”对 话 框 的 “数据 和 诊断 ”页 面 中 ， 双 击 “ 代 码 履 
六 ”所 在 行 ， 则 弹出 如 图 5.33 所 示 的 “代码 覆盖 明细 ”对 话 框 ， 在 其 中 可 以 选择 要 启用 代 
码 履 盖 分 析 的 项 目 ， 本 例 中 选择 BankDAL 项 目 。 


其 决 方案 目录 BenjaLLVbisvDebuf 
解决 方案 目录 》\BankDAL\bin\Debaz 
《解决 方案 目录 》\BanlalTestvbin\Debog 
和解 册 方 案 目录 BunkEntity\bin\Debue 
< 解 块 方案 目录 》\BankfebTest\binDebuz 


图 5.33 ”代码 覆盖 明细 
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(4) 设置 完成 代码 覆盖 以 后 ， 选 择 几 个 测试 并 运行 ， 测 试 运行 完成 后 在 测试 结果 试图 
中 右 击 ， 从 弹出 菜单 中 选择 “代码 覆盖 率 结果 ”， 如 图 5.34 所 示 。 


于 靶 | 的 |Aaministratereswsrmro 2010- -| 加 运行 ~ 调式 -是 了 | 直到 < 号 了 | 


测试 运行 已 完成 结果 : 2/2 个 通过 ; 选中 的 项 : 0 


本 停止 则 运行 名 ) 
查看 测试 结果 详细 信息 o) 
查看 运行 
打开 测试 


创建 性 能 会 话 @) 


总 复制 四 
全 选 多 

团 生 加 / 移 除 列 @) 
1 结果 


图 5.34 查看 代码 覆盖 率 结果 


(5) 代码 覆盖 率 结果 显示 如 图 5.35 所 示 ， 其 中 列 出 了 被 测试 项 目 中 所 有 类 和 方法 ， 以 
及 测试 覆盖 到 和 未 履 盖 到 的 代码 数量 和 比例 。 根 据 这 个 结果 ， 测 试 人 员 可 以 有 针对 性 地 调 
整 自己 的 测试 代码 ， 提 高 代码 覆盖 率 。 


Ee BanNAL all 
{3DAL 
有 Denkheeonthl 
EE CostomerDl 


0 
0 CT rc  | | 


图 种 PolicyDdl 
日 各 Sqlhelper 


ammasaag 


图 5.35 ”代码 覆 盖 率 结果 


(6) Visual Studio 还 支持 以 不 同 
颜色 显示 测试 覆盖 到 和 未 覆盖 到 的 代 
码 ， 让 开发 人 员 对 代码 覆盖 结果 一 目 
了 然 。 例 如 ， 在 图 5.35 中 ， 看 到 
CustomerDal 类 的 代码 覆盖 率 为 
42.31%， 如 果 想 查看 具体 哪些 代码 没 
有 和 获 盖 , 则 在 图 5.35 所 示 的 代码 覆盖 
视图 中 双击 CustomerDal 类 ， 会 打开 
CustomerDal.cs 文件 ， 并 分 别 用 浅 蓝 
色 和 浅 棕色 背景 显示 覆盖 到 和 未 覆盖 
到 的 代码 ， 如 图 5.36 所 示 。 


图 5.36 代码 覆盖 详情 
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5.5 Web 测试 


Visual Studio 不 但 支持 单元 测试 ， 还 支持 其 他 类 型 的 测试 ， 如 Web 测试 、 负 载 测试 、 
数据 库 单元 测试 等 。Web 测试 可 以 模拟 用 户 在 浏览 器 中 对 Web 应 用 程序 的 操作 ， 从 而 测试 
Web 应 用 程序 的 功能 和 性 能 。Web 测试 通过 录制 回放 的 方式 运行 ， 即 先 录 制 用 户 在 浏览 器 
中 的 操作 ， 然 后 再 重复 执行 用 户 的 操作 。 

当 用 户 输入 一 个 网 址 浏览 网 页 时 ， 会 向 Web 服务 器 发 送 一 个 HTTP Get 请 求 ， 服 务 器 
则 响应 此 请 求 返回 一 个 HTML 文件 ， 浏 览 器 以 可 视 化 的 方式 显示 出 HTML 文件 ， 用 户 就 
看 到 了 页 面 。 如 果 用 户 在 页 面 中 输入 数据 ， 并 且 通 过 提交 按钮 将 数据 回 发 给 服务 器 时 ， 会 
产生 一 个 HTTP Post 请 求 。Web 测试 的 原理 就 是 记录 用 户 操作 过 程 中 产生 的 HTTP 请 求 ， 
然后 以 编程 的 方式 向 Web 服务 器 模拟 发 送 这 些 请 求 ， 并 接收 和 验证 服务 器 返回 结果 。 下面 
以 测试 银行 转账 程序 BankApp 为 例 ， 说 明 Web 测试 的 方法 和 步骤 。 


名 提示: 由 于 Web 测试 需要 记录 用 户 打 开 的 URL， 并 模拟 用 户 操作 向 这 些 URL 请 求 或 者 
回 发 数据 ， 因 此 要 求 用 户 使 用 的 URL 是 固定 的 ， 而 Visual Studio 开发 时 使 用 的 
内 置 服务 器 会 随时 启动 和 停止 , 而且 端口 号 是 动态 生成 的 , 不 是 一 个 固定 的 URL， 
因此 Web 测试 时 不 宜 使 用 Visaul Studio 的 内 置 服务 器 ， 而 应 该 将 Web 项 目 发 布 
到 IIS 中 或 者 其 他 永久 的 Web 服务 器 上 。 


(1) 首先 需要 发 布 BankApp 项 目 。 在 解决 方案 资源 管理 器 中 ， 右 击 BankApp 项 目 名 
称 , 在 弹出 的 快捷 菜单 中 选择 “发 布 ”选项 , 则 弹出 如 图 5.37 所 示 的 “发 布 Web” 对 话 框 。 

(2) 在 其 中 ， 可 以 直接 将 Web 应 用 发 布 到 Web 服务 器 上 ， 也 可 以 发 布 到 本 地 文件 系 
统 上 。 本 例 将 BankApp 发 布 到 本 地 。 从 图 5.37 所 示 对 话 框 中 间 的 “发 布 方法 ”下 拉 列 表 
框 中 选择 “文件 系统 ”选项 ， 则 对 话 框 如 图 5.38 所 示 。 

(3) 在 对 话 框 中 单 击 右 下 方 的 省 略 号 按钮 ， 则 弹出 如 图 5.39 所 示 的 “目标 位 置 ”对 话 
框 ， 在 对 话 框 左 侧 选择 “本 地 IS”， 然 后 单 击 右上 角 的 创建 新 的 Web 应 用 按钮 鄙 以 创建 
Web 应 用 ,将 新 应 用 命名 为 BankWeb。 最 后 单 击 “ 打 开 ” 按 钮 ， 则 图 5.39 所 示 的 对 话 框 关 
闭 ， 回 到 图 5.38 所 示 对 话 框 中 。 单 击 “ 发 布 ”按钮 ， 则 BankApp 项 目 会 发 布 为 指定 的 IS 
项 目 ， 其 地 址 为 http://localhost/BankWeb。 

(4) 接 下 来 要 基于 http://localhost/BankWeb 创建 Web 测试 。 在 Visual Studio 中 ， 从 菜 
单 中 选择 “测试 ”|“ 新 建 测试 ”命令 ， 则 弹出 “添加 新 测试 ”对 话 框 。 在 对 话 框 的 测试 模 
板 列表 中 选择 “Web 性 能 测试 ”， 并 从 对 话 框 最 下 方 项 目 列表 中 选择 “创建 新 的 C# 测 试 项 
目 ”， 并 将 新 项 目 命名 为 BankWebTest， 如 图 5.40 所 示 。 

(5) 单 击 “ 确 定 ” 按 钮 ， 则 在 解决 方案 中 添加 了 一 个 新 的 测试 项 目 BankWebTest。 在 
项 目 上 右 击 ， 从 弹出 的 快捷 菜单 中 选择 “添加 ”|“Web 测试 ”命令 ， 则 会 自动 打开 一 个 浏 
览 器 窗口 ， 窗 口 左 侧 有 录制 、 停 止 录 制 等 按钮 ， 如 图 5.41 所 示 。 
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发 布 Yeb 


发 布 配置 文件 (H) 加 
ET 重 命名 邓 ) 


发 布 使 用 hs “打包 /发 布 Web” 和 “打包 /发 布 
SQL” 选项 卡 中 的 设置 ， 


查找 支持 一 溃 式 发 布 的 Web 承载 提供 程序。 

发布 
生成 配置 : Debug 
使 用 生成 配 慎 各 尘 基 来 更 改 配 司 


7 洲 上 JE] 
服务 VRL GD : [http: /ocalhost/Baneb 回 


例如 localhost 或 https://Remotes 


网 站 /应 用 程序 If); 『[ 


例如， 拒 认 网 站 /Wrhpp 或 JIomain < 


发 布 Yeb 


发 布 配置 文件 (H) 国 
Profilel sa ® A | ®) 


发 布 使 用 a “打包 /发 布 Web” 和 “打包 /发 布 


SR 选项 下 中 
| 有 
太保 留 目标 上 的 多 余 文 件 Ff 发布 
| 凭据 生成 配置 : Debug 
厂 允许 不 受信 任 的 证 书 全 用 生成 配 避 生 埋头 更 改 配 于 
忆 桂 此 选中 用 于 腕 信任 的 中 务 关 发 1 法 Wi [xs 吕 
用 尸 名 吕 目标 位 置 | Ea 
密码 遇 [Eee 用 本 地 副本 芝 换 匹配 的 文件 到 ) 
厂 保存 亿 码 g) 发 布 前 贡 除 所 有 现 有 文件 4) 


2 | 
图 5.37 发 布 Web 应 用 程序 图 5.38 选择 发 布 到 文件 系统 
ICE 3 
芭 本 地 Internet Information Server 
I Er elE 
i en Yo 仁 也 


ET 


误 试 名 称 0D BandebTest. webtest 
厂 使 用 安全 在 接 字 县 QU) 添加 到 | 试 项 目 @)。 [全 建新 的 Vi suwl C# 测试 项 目 | 
| 
图 5.39 ”发布 到 本 地 IS 图 5.40 新 建 测试 


(6) 在 图 5.41 所 示 的 Web 测试 录制 窗口 中 ， 启 动 录制 后 ， 用 户 在 浏览 器 中 所 有 操作 
都 会 被 记录 下 来 。 为 了 测试 银行 转账 程序 ， 在 浏览 器 地 址 栏 中 输入 银行 转账 的 URL， 可 以 
看 到 ， 左 侧 的 录制 窗口 中 记录 下 了 用 户 的 操作 ， 产 生 了 新 的 数据 ， 如 图 5.42 所 示 。 

(7) 在 图 5.42 所 示 窗 口中 执行 其 他 需要 测试 的 操作 ， 此 处 为 输入 转账 信息 ， 然 后 单 击 
转账 按钮 进行 转账 。 可 以 先 故意 输入 错误 数据 , 然后 再 输入 正确 数据 ， 加强 对 程序 的 测试 。 
在 浏览 器 中 这 些 操作 都 被 录制 下 来 。 
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ET 本 


从- 有 目 - 本 本- 中 - 安 昌 - I 上 AO- -+-” 


一 属 
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CT te 


BiS 可 到 OO 共 SECOR 
So 3 旦 -最 是 - 斌 习 癌 -|s 要 


突 收藏 大。 忽 良 行 转账 程序 ( 三 层 结构 版 ) 
Web Test Recorder x 


[Becord NW Passe asto | 了 | X 


四 
四 http://localhest:2443/favicon ice 


转 出 账号 ， 转账 全 额 ， 


| 
接收 账号 账号 名 称 ， # 
[im -及 


三 三 厂矿 厂矿 网 坝 Be FS 


图 5.42 用户 操作 被 录制 


(8) 录制 完成 后 ， 单 击 “ 停 止 录 制 ” 按 钮 ， 则 录制 浏览 器 关闭 ， 回 到 Visual Studio 中 ， 
并 且 显 示 了 刚才 录制 的 脚本 ， 如 图 5.43 所 示 。 


(http://localhost:2443/TierBankF age. aspx 
Hhttp: /flocalhost:2443/TierBankPage. aspx 
: | 入 http: /localhost:2443/ favicon. ico 
- 证 规则 
IE Response URL 
导 Response Time Goal 


图 5.43 录制 的 脚本 


(9) 用 户 对 录制 的 脚本 进行 编辑 。 如 果 要 修改 用 户 输入 的 目的 账号 ， 则 在 Web 测试 视 
图 中 展开 单 击 “ 转 账 ”按钮 时 所 产生 的 HITP 请 求 ， 并 从 表单 回 发 参数 中 选择 目的 账号 所 
对 应 的 表单 字段 destination， 即 可 在 右 侧 属 性 窗口 中 修改 此 参数 的 值 ， 如 图 5.44 所 示 。 
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鹿 舍 下 加 久 寺 上 W 尘 果 加 Ws0z 印 吾 世 续 于 ! /名 玫 基 和 司 千林 


5.44 ”编辑 录制 的 脚本 


(10) 当 用 户 执行 一 个 Web 程序 时 ， 可 以 根据 页 面 上 显示 的 内 容 判 断 程序 是 否 运行 成 
功 。 在 进行 Web 测试 时 , 可 以 通过 添加 规则 让 测试 程序 根据 页 面 上 的 内 容 进 行 判断 。 例如， 
在 银行 转账 程序 中 ， 如 果 用 户 输入 错误 的 账号 ， 则 页 面 上 应 该 显示 “顾客 不 拥有 所 请 求 账 
号 ”的 错误 提示 。 为 Web 测试 添加 验证 规则 的 操作 步骤 为 ， 在 图 5.43 所 示 的 Web 测试 视 
图 中 ， 右 击 “ 验 证 规则 ”文件 夹 ， 从 弹出 的 快捷 菜单 中 选择 “添加 验证 规则 ”命令 ， 则 弹 
出 如 图 5.44 所 示 的 对 话 框 。 

在 图 5.45 所 示 的 “添加 验证 规则 ”对 话 框 左 侧 列 出 了 一 些 常用 验证 规则 ， 如 验证 某 个 
HTML 标记 中 的 值 、 验 证 表单 中 某 个 字段 、 验 证 最 大 请 求 时 间 、 验 证 页 面 上 出 现 的 文本 等 。 
在 本 例 中 , 如 果 用 户 输入 错误 的 账号 , 应 该 给 出 错误 提示 , 所 以 从 验证 规则 列表 中 选择 “ 查 
找 文 本 ”， 然 后 从 右 侧 属性 窗口 中 设置 查找 “顾客 不 拥有 所 请 求 账号 ”， 并 且 设置 如 果 找 
到 文件 则 通过 测试 。 


图 5.45 添加 验证 规则 


(11) 在 测试 视图 中 单 击 “ 运 行 测试 ”按钮 ， 则 Visual Studio 会 回放 刚才 所 录制 的 用 户 
操作 。 在 Visual Studio 窗口 中 显示 了 Web 程序 运行 界面 和 每 次 URL 请 求 执行 情况 ， 如 是 
否 成 功 、 响 应 时 间 等 ， 如 图 5.46 所 示 。 
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efor 3 0 LAN Blit ram settin 
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tp /flocelhost: 249/Tier BanlP ege. asp 200 OK 0.249 sec 
| http /lechort:203/ Earicon ico 404 Hot Found 0.058 see 
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银行 转账 程序 


客户 编号 , [110101198001011234 “登录 密码 : 「 支付 密码 ,[ 
转 出 账号，esf 转账 全 新 [50600 


接收 账号 ，[235645 账号 名 称 ， B00 bd 
顾客 不 拥有 所 请 求 联 户 


图 5.46 Web 测试 运行 界面 


(12) 如 前 所 述 ，Web 测试 的 原理 就 是 记录 用 户 操作 过 程 中 产生 的 HTTP 请 求 ， 然 后 
以 编程 的 方式 向 Web 服务 器 模拟 发 送 这 些 请 求 ， 并 接收 和 验证 服务 器 返回 结果 。 这 些 步 骤 
都 对 应 于 一 定 的 代码 ， 通 过 保存 和 手工 编辑 这 些 代 码 ， 用 户 可 以 实现 更 加 个 性 化 的 需求 
要 查看 Web 测试 所 对 应 的 代码 ， 在 Web 测试 视图 中 ， 右 击 Web 测试 名 称 ， 从 弹出 菜单 中 
选择 “生成 代码 ”命令 。 本 例 所 生成 的 关键 代码 如 下 。 


// 创 建 第 一 个 HTTP Get 请 求 ， 显 示 银 行 转账 页 面 

WebTestRequest request1l1 = new WebTestRequest ("http://localhost:2443 
/TierBankPage .aspx") ; 

Fequest1l .ThinkTime = 77; 

ExtractHiddenFields extractionRulel = new ExtractHiddenFields () 7 
extractionRulel .Required = true; 

extractionRulel .HtmlDecode = true; 

extractionRulel .ContextParameterName = "1"; 


requestl1.ExtractValues += new EventHandler<ExtractionEventArgs> 
(extractionRulel .Extract); 

yield return requestl1; 

requestl1 = null; 

// 创 建 第 二 个 HTTP 请 求 ， 为 一 个 Post 请 求 ， 向 服务 器 提交 转账 信息 

WebTestRequest request2 = new WebTestRequest ("http://localhost:2443 
/TierBankPage.aspx"); 

request2.Method = "POST"; // 设 置 为 POST 方法 
FormPostHttpBody request2Body = new FormPostHttpBody(); 

// 以 下 语句 添加 Post 的 各 个 参数 
Tedquest2Body.FormPostParameters.Add ("customer", "110101198001011234"); 
request2Body .FormPostParameters.Add ("loginPass", "123456"); 
request2Body .FormPostParameters.Add ("payPass", "12345600"); 
request2Body .FormPostParameters.Add ("source", "account0000"); 
request2Body .FormPostParameters.Add ("money", "50"); 

request2Body .FormPostParameters.Add ("destination", "TestAccount002"); 
request2Body .FormPostParameters.Add ("destinationName",， "龙套 张 "); 
request2Body .FormPostParameters.Add ("ok", "转账 "); 

request2.Body = request2Body; 

yield return request2; 

request2 = null; 
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5.6 负载 测试 


在 进行 软件 测试 时 ， 大 多 数 情况 下 都 需要 测试 软件 在 多 用 户 并 发 环境 下 的 运行 情况 ， 
包括 性 能 、 响 应 时 间 、 稳 定性 等 。Visual Studio 中 的 负载 测试 可 以 模拟 多 用 户 并 发 执行 的 
环境 ， 满 足以 上 测试 要 求 。 负 载 测试 由 一 系列 Web 测试 或 单元 测试 组 成 ， 这 些 测试 在 存在 
多 个 模拟 用 户 的 情况 下 运行 一 段 时 间 。 负 载 测 试 可 以 用 于 以 下 几 种 不 同 的 测试 类 型 。 

口 冒 烟 测 试 : 确定 在 短 时 间 内 负载 较 小 时 应 用 程序 如 何 执行 。 

口 压力 测试 : 确定 在 较 长 时 间 内 负载 较 大 时 应 用 程序 是 否 能 成 功 运行 。 

口 性 能 测试 : 确定 应 用 程序 的 响应 能 力 。 

口 容量 计划 测试 : 确定 在 各 种 容量 下 应 用 程序 如 何 执行 。 

下 面 以 银行 转账 程序 测试 为 例 ， 说 明 Visual Studio 中 进行 负载 测试 的 步骤 。 

(1) 打开 银行 转账 程序 ， 从 菜单 中 选择 “测试 ”|“ 新 建 测试 ”命令 ,在 打开 的 “新 建 
测试 ”对 话 框 中 ， 选 择 “ 负 载 测 试 ”， 则 会 显示 “新 建 负载 测试 向 导 ” 欢 迎 对 话 框 。 在 其 
中 单 击 “ 下 一 步 ” 按 钮 ， 进 入 “负载 方案 设置 ”对 话 框 ， 在 其 中 可 以 设置 方案 名 称 和 用 户 
思考 时 间 ， 如 图 5.47 所 示 。 本 例 的 思考 时 间 采 用 以 记录 的 思考 时 间 为 中 心 的 正 态 分 布 。 
全 提示 : 思考 时 间 用 于 模拟 人 类 与 网 站 执行 的 各 种 交互 之 间 存 在 等 待 时 间 这 种 行为 。Web 

测试 中 的 各 个 请 求 之 间 及 负载 测试 方案 的 各 个 测试 迭代 之 间 均 会 产生 思考 时 间 。 
在 负载 测试 中 使 用 思考 时 间 对 于 创建 更 为 精确 的 负载 模拟 很 有 用 。 

(2)“ 负 载 测试 向 导 ” 的 下 一 步骤 为 设置 负载 模式 ， 可 以 选择 固定 的 用 户 数 或 者 分 级 

负载 。 本 例 选择 固定 50 个 用 户 数 ， 如 图 5.48 所 示 。 


ss | 编 扯 负 夫 而 区 方 安 的 语 旭 
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月 的 人 人 二 于 的 各 称 ); 
JE 
生 者 梳 式 时 关 时 间 孔 轨 妆 这 
i 全 模 蜡 ER 选择 模拟 负载 的 负载 模式 : 
Wd 全 用 以 记 和 到 寺 时 间 为 中 心 扣 下放 分 UD 个 常量 负载 (人); 
各 招生 FERED) 
a 用 Pi 数 四 ， 『 所 当 用 记 


计划 着 
eee 人 之 的 加 是 相 0)。 [ID 当 秒 个 分 颁 负 载 G) ; 


开始 用 Pi 计数 四 : 「 0 习 用 记 
单 步 持续 时 间 Wm): | 10 习 种 
单 步 用 户 计 数 罗 :| 00 习 用 户 数 /步骤 


su TS] zx | mm 是 大 用 户 计数 是: | 9 习 用 记 
图 5.47 设置 测试 方案 图 5.48 设置 负载 模式 


(3) 在 “负载 测试 向 导 ” 中 单 击 “ 下 一 步 ” 按 钮 ， 直 到 进入 如 图 5.49 所 示 的 测试 组 合 
步 又， 在 此 可 以 指定 负载 测试 中 包含 的 单元 测试 和 Web 测试 。 

(4) 在 图 5.49 所 示 对 话 框 中 单 击 “ 添 加 ”按钮 ， 则 打开 如 图 5.50 所 示 的 “添加 测试 ” 
对 话 框 。 对 话 框 左 侧 列 出 了 当前 项 目 中 存在 的 测试 , 右 侧 列 出 了 包含 在 负载 测试 中 的 测试 。 
从 左 侧 选择 需要 添加 的 测试 ， 单 击 对 话 框 中 间 的 “>” 按 钮 ， 将 其 添加 到 负载 测试 中 。 
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FEEOI 了 


加 已 加 0 个 测 工 ] 
插 一 个 或 多 个 测试 于 加 到 由 合并 指定 一 个 分 发 方案 吧 )- 
EE 四 区 = 3| mw... | 
单 击 “ 添 加 ”以 添加 测试 #2 | 
图 El 
| 0 
图 5.49 负载 测试 组 合 图 5.30 向 负载 测试 中 添加 测试 


(5) 在 “负载 测试 向 导 ” 的 下 一 步骤 中 ， 可 以 指定 测试 所 处 的 模拟 网 络 环境 。 可 以 选 
择 局 域 网 或 者 不 同 速度 的 广域网 的 组 合 为 模拟 网 络 环境 ， 如 图 5.51 所 示 。 
(6) 在 “负载 测试 向 导 ” 的 下 一 步骤 中 ， 可 以 为 测试 指定 不 同 的 浏览 器 组 合 ， 例 如 ， 


可 以 使 用 FireFox、Intemet Explorer、Netscape 等 不 同 的 浏览 器 对 银行 转账 程序 进行 测试 ， 
如 图 5.52 所 示 。 
将 一 个 或 多 个 浏览 器 奖 弄 添加 到 组 合并 指定 一 个 分 发 方案 他); 
区 器 类 型 EE 3| mw | 
iretox 3.0 到 25 厂 | 和 的) 
了 全 | - 口 分 发 四) 
25 口 
E33 . D 
号 歼 | 100 al FE 
图 5.51 配置 模拟 网 络 环境 图 5.52 浏览 器 组 合 


(7) 在 “负载 测试 向 导 ” 最 后 一 个 步骤 中 ， 可 以 指定 测试 持续 运行 的 时 间或 者 重复 执 
行 的 次 数 ， 如 图 5.53 所 示 。 
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个 测试 法 代 (I) 
测试 渤 代 瑟 ) 19 习 
详细 信息 


采样 速率 人 @): 5 当 物 


景 多 错误 详细 信息 员 ) ; 10 习 
验证 级 别 QD : 低 - 调用 标记 为 低 的 验证 规则 到 


图 5.53 负载 测试 运行 设置 
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(8) 负载 测试 创建 完成 后 ， 单 击 工具 栏 上 的 运行 测试 按钮 以 运行 该 测试 。 测 试 运行 过 
程 中 ， 将 以 图 表 方 式 实 时 显示 各 个 数据 指标 ， 如 图 5.54 所 示 。 
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图 5.54 负载 测试 运行 监控 


27， 外 结 


本 章 主 要 介绍 了 3 部 分 内 容 : 源码 管理 、 三 层 结构 和 软件 测试 。 首 先 介绍 了 Visual 
SourceSafe 源码 管理 工具 的 使 用 及 其 与 Visual Studio 的 集成 ， 接 下 来 介绍 了 三 层 结构 并 分 
析 其 优势 ， 最 后 介绍 了 几 种 常用 的 软件 测试 : 单元 测试 、Web 测试 和 负载 测试 。 
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现在 互联 网 已 经 成 为 人 们 日 常生 活 中 一 个 重要 的 组 成 部 分 ， 人 们 上 网 看 新 闻 ， 在 网 上 
选 商 店 ， 在 论坛 中 讨论 问题 等 。 广 大 企业 和 商家 也 充分 意识 到 了 互联 网 平台 的 重要 性 ， 纷 
纷 建 立 企业 网 站 ， 发 布 供求 信息 。 企 业 网 站 建立 以 后 ， 如 何 让 更 多 的 用 户 来 访问 是 一 个 非 
常 重要 的 问题 。 

在 当今 互联 网 上 ， 用 户 查 找 某 种 资料 的 方式 一 般 都 是 使 用 搜索 引擎 进行 搜索 ， 然 后 根 
据 搜索 引擎 列 出 的 搜索 结果 和 链接 访问 相关 网 站 。 这 就 意味 着 , 在 搜索 引擎 的 搜索 结果 中 
排名 越 靠 前 的 网 站 被 访问 到 的 机 率 越 高 ， 而 更 高 的 访问 次 数 也 就 意味 着 更 多 的 商品 销售 和 
利润 。 如 何 才 能 使 自己 的 网 站 在 搜索 引擎 的 搜索 结果 中 排名 靠 前 呢 ? 这 对 企业 来 说 是 一 个 
很 重要 的 问题 ， 这 也 就 是 搜索 引擎 优化 所 要 解决 的 问题 。 


6.1 搜索 引擎 优化 简介 


搜索 引擎 优化 对 应 的 英文 名 称 是 Search Engine Optimization， 简 称 SEO， 是 近年 来 较 
为 热门 的 一 个 技术 领域 。 本 节 将 介绍 搜索 引擎 优化 的 概念 、 动 机 和 原理 ， 使 读者 对 搜索 引 
擎 优化 建立 一 个 印象 。 


6.1.1 搜索 引擎 优化 基本 概念 


随 着 互联 网 的 普及 ， 越 来 越 多 的 企业 和 商家 建立 自己 的 网 站 ， 在 互联 网 上 发 布 自己 的 
产品 信息 , 甚至 有 很 多 个 人 也 建立 了 个 人 网 站 或 者 博客 。 作 为 一 个 展示 产品 的 平台 和 媒介 ， 
相对 于 传统 的 产品 宣传 方式 来 说 ， 互 联网 拥有 更 丰富 的 内 容 、 更 广泛 的 受众 和 更 廉价 的 成 
本 ， 因 而 也 越 来 越 得 到 企业 和 商家 的 重视 。 

对 于 一 个 企业 来 说 ， 为 了 达到 宣传 企业 商品 的 目的 ， 建 立 企业 网 站 只 是 第 一 步 。 网 站 
建立 后 ， 如 果 没 有 顾客 访问 ， 则 起 不 到 任何 作用 。 企 业 要 想 提 高 网 站 的 访问 量 ， 可 以 有 两 
种 主要 途径 ， 第 一 是 为 网 站 做 广告 ， 这 种 方式 成 本 较 高 ;第 二 就 是 搜索 引擎 优化 ， 这 种 方 
式 成 本 较 低 且 效 果 明 显 。 

所 谓 搜索 引擎 优化 ， 是 指 通过 总 结 搜索 引擎 的 排名 规律 ， 对 网 站 进行 合理 优化 ， 使 目 
标 网 站 在 搜索 引擎 〈 如 Google 和 百度 ) 的 排名 提高 ， 从 而 获得 更 多 的 用 户 访问 量 。 搜 索引 
擎 优化 技术 的 出 现 与 搜索 引擎 的 广泛 应 用 密 不 可 分 。 现 在 人 们 要 从 网 上 查找 某 些 信息 时 ， 
通常 都 会 通过 搜索 引擎 针对 某 关键 字 进行 搜索 ， 然 后 再 从 搜索 引擎 的 搜索 结果 中 点 击 链接 
进入 相应 网 站 。 

由 于 互联 网 的 海量 信息 量 ， 对 于 一 个 关键 字 的 搜索 通常 会 有 庞大 的 搜索 结果 ， 搜 索引 
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擎 以 分 页 的 方式 显示 这 些 结果 。 显 然 列 在 第 一 页 的 搜索 结果 会 有 最 大 的 机 率 得 到 访问 ， 而 
排列 在 第 几 十 、 几 百 、 甚 至 几 千 页 以 后 的 搜索 结果 几乎 不 会 被 访问 。 例 如 ， 在 Google 中 搜 
索 “ 烤 箱 "， 则 得 到 获得 约 4 830 000 条 结果 ， 若 每 页 显示 10 条 结果 ， 则 一 共有 48 万 页 。 
对 于 一 个 生产 烤箱 的 企业 来 说 ， 当 然 希望 自己 厂家 的 网 站 能 够 显示 在 搜索 结果 第 一 页 甚至 
第 一 条 的 位 置 ， 而 不 希望 在 位 于 搜索 结果 的 几 十 页 以 后 。 

如 何 才能 使 自己 的 网 站 在 搜索 结果 中 排名 靠 前 呢 ? 这 就 需要 理解 和 掌握 搜索 引擎 的 
工作 原理 和 对 搜索 结果 进行 排名 的 算法 ， 并 且 以 此 为 根据 有 针对 性 的 完善 自己 的 网 站 ， 从 
而 提高 自己 网 站 的 搜索 引擎 排名 。 


6.1.2 搜索 引擎 工作 原理 


搜索 引擎 按 其 工作 方式 主要 分 为 3 种 , 分 别 是 全 文 搜索 引擎 (Full Text Search Engine)、 
目录 索引 类 搜索 引擎 〈Search Index/Directory) 和 元 搜索 引擎 (Meta Search Engine )。 

全 文 搜索 引擎 从 互联 网 上 提出 各 个 网 站 和 页 面 信息 ， 并 把 这 些 信 息 处 理 后 保存 到 数据 
库 中 ， 用 户 使 用 搜索 引擎 进行 搜索 时 ， 从 数据 库 中 检索 与 用 户 查询 条 件 匹配 的 记录 ， 并 按 
照 一 定 的 条 件 对 查询 结果 进行 排序 后 将 结果 返回 给 用 户 。 全 文 搜索 引擎 是 真正 意义 上 的 搜 
索引 擎 ,比较 典型 的 有 国外 的 Google 和 国内 的 百度 。 本 章 所 讲 的 搜索 引擎 优化 主要 针对 此 
类 搜索 引擎 。 

目录 索引 虽然 有 搜索 功能 ， 但 在 严格 意义 上 讲 算 不 上 是 真正 的 搜索 引擎 ， 仅 仅 是 按 目 
录 分 类 的 网 站 链接 列表 而 已 。 目 录 搜 索引 擎 收录 了 互联 网 上 存在 的 网 站 ， 给 每 个 网 站 分 配 
若干 个 标签 ， 用户 可 以 根据 标签 找到 相关 网 站 。 例 如 ,用户 可 以 通过 “编程 `“ 游 戏 "“ 新 
闻 ” 等 标签 找到 相关 的 网 站 。 比 较 典 型 的 目录 索引 有 雅虎 、 搜 狐 等 。 

元 搜索 引擎 是 建立 在 其 他 搜索 引擎 基础 上 的 搜索 引擎 。 当 用 户 在 元 搜索 引擎 中 进行 查 
询 时 ， 元 搜索 引擎 把 查询 条 件 传递 给 其 他 多 个 搜索 引擎 ， 使 用 其 他 搜索 引擎 进行 搜索 ， 然 
后 将 查询 结果 返回 给 用 户 。 比 较 典 型 的 元 搜索 引擎 有 MetaSearch (http://metasearch.com/)、 
疯 搜 (http://www.metasoo.com/) 等 。 

鉴于 目前 主流 的 搜索 引擎 为 全 文 搜索 引擎 ， 本 章 将 以 这 种 搜索 引擎 为 例 说 明 搜 索引 擎 
工作 原理 和 优化 技术 。 当 用 户 使 用 搜索 引擎 进行 搜索 时 , 并 不 是 实时 在 互联 网 上 进行 搜索 ， 
而 是 在 一 个 预先 整理 好 的 网 页 索引 数据 库 中 进行 搜索 ,然后 将 搜索 结果 排序 并 显示 给 用 户 。 
搜索 引擎 的 工作 原理 包含 3 个 步骤 : 从 互联 网 上 抓 取 网 页 建立 索引 数据 库 ， 在 索引 数据 
库 中 搜索 排序 。 


1. 抓 取 网 页 


页 面 抓 取 是 指 从 互联 网 上 不 断 访问 页 面 并 取 回 页 面 内 容 的 过 程 ， 这 一 工作 是 搜索 引擎 
最 基础 的 工作 。 互 联网 上 的 页 面 是 由 HTML 文件 组 成 的 ， 这 些 HTML 文件 相互 链接 ， 形 
成 一 个 网 状 结构 。 搜 索引 擎 的 页 面 抓 取 程序 通过 这 些 链接 从 一 个 页 面 到 达 另 外 一 个 页 面 ， 
好 像 一 个 蜘蛛 在 页 面 组 成 的 网 上 疏 来 聆 去 ， 因 此 搜索 引擎 的 页 面 抓 取 程 序 通常 被 形象 的 称 
为 “ 蜂 蛛 〈Spider)” 或 者 “ 疏 虫 CCrawler) ”。 
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2. 建立 索引 数据 库 


搜索 引擎 蜘蛛 把 网 页 数据 抓 取 回 来 以 后 ， 需 要 对 这 些 内 容 进行 分 析 整 理 ， 提 取 相 关 网 
页 信息 (包括 网 页 所 在 URL、 编 码 类 型 、 页 面 内 容 包 含 的 关键 词 、 关 键 词 位 置 、 生 成 时 间 、 
与 其 他 网 页 的 链接 关系 等 ) 根据 一 定 的 相关 度 算法 进行 大 量 复杂 计算 , 得 到 每 一 个 网 页 针 
对 页 面 内 容 中 及 超 链 中 每 一 个 关键 词 的 相关 度 (或 重要 性 ), 然后 用 这 些 相关 信息 保存 到 一 
个 特定 结构 的 索引 数据 库 中 。 


3. 搜索 和 排序 


当 用 户 输入 关键 词 搜索 后 ， 由 搜索 系统 程序 从 网 页 索引 数据 库 中 找到 符合 该 关键 词 的 
所 有 相关 网 页 。 因 为 所 有 相关 网 页 针对 该 关键 词 的 相关 度 早已 算 好 ， 所 以 只 需 按照 现成 的 
相关 度数 值 排序 ， 相 关 度 越 高 ， 排 名 越 靠 前 。 最 后 ， 由 页 面 生成 系统 将 搜索 结果 的 链接 地 
址 和 页 面 内 容 摘要 等 内 容 组 织 起 来 返回 给 用 户 。 

搜索 引擎 蜘蛛 一 般 要 定期 重新 访问 所 有 网 页 〈 各 搜索 引擎 的 周期 不 同 ， 可 能 是 几 天 、 
几 周 或 几 个 月 ， 也 可 能 对 不 同 重要 性 的 网 页 有 不 同 的 更 新 频率 ), 更 新 网 页 索引 数据 库 ， 以 
反映 出 网 页 内 容 的 更 新 情况 ， 增 加 新 的 网 页 信息 ， 去 除 死 链接 ， 并 根据 网 页 内 容 和 链接 关 
系 的 变化 重新 排序 。 这 样 ， 网 页 的 具体 内 容 和 变化 情况 就 会 反映 到 用 户 查 询 的 结果 中 。 


6.1.3 ”搜索 引擎 排名 因素 


搜索 引擎 排名 算法 非常 复杂 ， 要 综合 考虑 许多 因素 形成 对 一 个 页 面 的 评价 。 本 节 将 介 
绍 影响 搜索 引擎 排名 的 主要 因素 ， 搜 索引 擎 优化 就 是 针对 这 些 因素 完善 页 面 ， 使 搜索 引擎 
对 页 面 的 评价 增加 ， 从 而 提高 在 搜索 结果 中 的 排名 。 影 响 一 个 页 面 在 搜索 引擎 中 排名 的 因 
素 可 以 分 为 两 大 类 : 页 面 内 部 因素 和 页 面 外 部 因素 。 页 面 内 部 因素 包括 以 下 内 容 : 

1. 页 面 标题 

页 面 标题 即 HTML 文档 中 <title> 标 记 中 的 内 容 。 页 面 标题 应 该 明确 说 明 页 面 的 内 容 ， 
最 好 包含 搜索 关键 字 。 给 页 面 指定 一 个 合适 的 标题 能 够 大 大 提高 搜索 引擎 对 页 面 的 评价 值 。 
页 面 标题 应 该 与 页 面 内 容 密切 相关 ， 不 能 使 用 太 笼 统 的 标题 ， 更 不 能 使 标题 与 内 容 无 关 。 
一 个 网 站 中 要 避免 所 有 页 面具 有 完全 相同 的 标题 。 

2. 正文 标题 

此 处 使 用 正文 标题 是 为 了 与 页 面 标题 相 区 别 。 页 面 标题 是 指 HTML 文档 中 <title> 标 记 
中 的 内 容 ， 在 浏览 器 中 显示 于 浏览 器 窗口 标题 栏 。 而 正文 标题 是 指 HTML 文档 中 <h1>、 
<h2> 等 标记 内 容 。 通 常 这 些 标题 说 明了 页 面 的 主要 内 容 和 大 纲 ， 对 搜索 引擎 来 说 ， 这 些 标 
题 具 有 比 正文 更 大 的 权重 。 如 果 页 面 的 正文 标题 中 包含 与 搜索 关键 字 密 切 相 关 的 内 容 ， 则 
搜索 引擎 对 此 页 面 的 评价 更 高 。 


3. 页 面 内 容 
页 面 正 文中 应 该 包含 与 被 搜索 关键 字 相 关 的 文字 和 图 片 。 页 面 中 出 现 搜索 关键 字 的 密 
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度 越 高 ， 则 说 明 页 面 与 被 搜索 内 容 相关 性 越 高 ， 从 而 具有 更 好 的 搜索 引擎 评价 。 对 于 页 面 
中 的 图 片 要 设置 其 替换 文本 《〈 即 img 标记 中 的 alt 属性 )。 替 换文 本 的 作用 主要 有 两 个 ， 一 
方面 是 当 图 片 无 法 加 载 时 可 以 显示 替换 文本 作为 说 明 性 文字 ， 另 一 方面 是 由 于 搜索 引擎 目 
前 无 法 识别 图 片 内 容 ， 只 能 根据 替换 文本 来 判断 图 片 内 容 。 

影响 一 个 页 面 搜索 引擎 排名 的 外 部 因素 主要 包括 到 此 页 面 的 外 部 链接 ， 包 括 链 接 的 数 
量 、 相 关 性 、 链 接 文 本 等 。 一 个 页 面 被 越 多 外 部 链接 引用 ， 就 说 明 这 个 页 面 越 重 要 ， 搜 索 
引擎 对 其 评价 也 就 越 高 。 链 接 数 量 只 是 一 个 因素 ， 不 同 的 链接 其 质量 也 不 同 。 例 如 ， 一 个 
来 自 著 名 网 站 的 链接 和 一 个 来 自 不 知名 网 站 的 链接 其 价值 是 不 一 样 的 。 对 于 外 部 链接 还 应 
考虑 其 相关 性 ， 例 如 ， 如 果 页 面 a 是 一 个 讲 编程 知识 的 页 面 ， 那 么 在 链接 到 页 面 a 的 所 有 
链接 中 ， 如 果 大 部 分 都 来 自 编程 相关 的 页 面 ， 则 认为 这 些 链接 质量 较 好 ， 而 来 自 于 其 他 无 
关 行 业 页 面 的 链接 质量 就 不 好 。 

综 上 所 述 ， 如 果 一 个 页 面具 有 明确 的 与 内 容 相关 的 标题 、 页 面 内 容 与 内 容 密切 相关 、 
页 面 拥有 高 质量 的 外 部 链接 ， 那 么 这 个 页 面 在 搜索 引擎 中 就 具有 较 好 的 排名 。 页 面 标题 、 
内 容 、 外 部 链接 也 是 搜索 引擎 优化 的 主要 方向 ， 通 过 合理 的 设置 页 面 标题 和 内 容 ， 想 办 法 
让 其 他 页 面 引用 自己 的 页 面 ， 就 可 以 提高 自己 页 面 的 搜索 引擎 排名 。 


6.1.4 SEO 作 炊 


如 果 所 述 ， 通 过 设置 合适 页 面 标题 、 页 面 文本 、 图 片 替 换文 本 、 页 面 链接 等 ， 可 以 提 
高 页 面 的 搜索 引擎 排名 。 但 是 如 果 滥 用 这 些 因素 ， 甚 至 做 假 试 图 欺骗 搜索 引擎 ， 不 但 达 不 
到 预想 的 目的 ， 而 且 还 会 受到 搜索 引擎 的 惩罚 ， 被 降低 排名 甚至 从 搜索 引擎 结果 中 除名 。 
在 做 搜索 引擎 优化 时 ， 需 要 了 解 这 些 作 整 手段 ， 以 避免 无 意 中 使 用 受到 搜索 引擎 惩罚 。 搜 
索引 擎 优化 常用 作弊 手段 主要 有 以 下 几 种 。 


1. 关键 字 堆 砌 


为 了 增加 关键 词 的 出 现 频次 ， 故 意 在 网 页 代码 中 ， 如 在 meta、tile、 注 释 、 图 片 alt 等 
地 方 重复 书写 某 关键 词 的 行为 。 下 面 的 页 面 代码 是 一 个 关键 字 堆 砌 的 例子 ， 此 页 面 为 了 提 
高 ASPNET、C#、 案 例 、 教 程 等 关键 字 搜 索 排名 ， 在 页 面 中 添加 了 许多 关键 字 ， 而 这 些 关 
键 字 仅 仅 是 堆砌 而 已 ， 不 是 出 现在 正文 的 句子 中 ， 这 是 关键 字 堆 砌 作 浆 。 


<head> 

<title>ASP.NET 案例 教程 </title> 
</head> 
<body> 
<! 一 以 下 为 关键 字 堆 砌 --> 
ASP.NET C# jQuery Web Service SQL Server 编程 项 目 案例 教程 
ASP.NET C# jQuery Web Service SQL Server 编程 项 目 案例 教程 
ASP.NET C# jQuery Web Service SQL Server 编程 项 目 案例 教程 
ASP.NET C# jQuery Web Service SQL Server 编程 项 目 案例 教程 
ASP.NET C# jQuery Web Service SQL Server 编程 项 目 案例 教程 
<!-- 以 下 才 是 页 面 的 真正 内 容 〈 省 略 ) --> 
</body> 
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2. 虚假 关键 词 


通过 在 meta 或 tile 中 设置 与 网 站 内 容 无 关 的 关键 词 ， 如 设置 热门 关键 词 ， 以 达到 误 
户 进入 网 站 的 目的 。 同 样 的 情况 也 包括 链接 关键 词 与 实际 内 容 不 符 的 情况 。 
HTML 文档 中 的 <meta> 元 素 可 提供 有 关 页 面 的 元 信息 (meta-information)， 比 如 针对 
搜索 引擎 和 更 新 频 度 的 描述 和 关键 词 。meta 标签 必须 位 于 文档 的 头 部 ， 不 包含 任何 内 容 。 
meta 标签 的 属性 定义 了 与 文档 相关 联 的 名 称 / 值 对 ，meta 标签 与 搜索 引擎 相关 的 属性 主要 
有 以 下 两 个 。 

<meta name="keywords" content=" 搜 索引 擎 从 此 处 获得 页 面 的 关键 词 "/> 

<meta name="description" content=" 搜 索引 擎 从 此 处 获得 页 面 的 主要 内 容 "/> 

Meta 元 素 中 的 内 容 对 于 用 户 来 说 是 不 可 见 的 , 但 是 却 可 以 被 搜索 引擎 访问 到 。 如 上 代 
码 所 示 , 通过 在 meta 标记 中 添加 适当 内 容 , 可 以 通知 搜索 引擎 页 面 此 页 面 的 关键 词 和 主要 


导 


Nag 


现象 就 是 虚假 关键 词 作 次 。 以 下 是 一 个 虚假 关键 词 的 例子 ， 例 如 在 电影 阿 凡 达 刚 上 映 时 ， 
网 上 许多 人 都 在 搜索 阿 凡 达 ， 为 了 提高 页 面 被 搜 中 的 概率 ， 在 页 面 的 <meta> 中 添加 了 虚假 
的 关键 词 ， 试 图 欺骗 搜索 引擎 。 
<head> 
<meta name="keywords" content=" 阿 凡 达 ， 高 清 ， 电 影 "/> 
<meta name="description" content=" 电 影 阿 凡 达 高 清 在 线 观 看 "/> 
</head> 
<body> 


页 面 中 的 真实 内 容 ， 其 实 是 一 些 其 他 文字 ， 与 阿 凡 达 无 关 ， 更 不 能 在 线 观 看 
</body> 


3. 垃圾 链接 


“链接 工厂 ”( 也 称 为 “大 量 链接 机 制 ”) 指 由 大 量 网 页 交叉 链接 而 构成 的 一 个 网 络 系 
统 。 一 个 站 点 加 入 “链接 工厂 ”后 ， 一 方面 可 得 到 来 自 该 系统 中 所 有 网 页 的 链接 ， 同 时 作 
为 交换 需要 奉献 自己 的 链接 ， 籍 此 方法 来 提升 链接 得 分 。 

4. 隐形 文本 和 链接 


为 了 增加 关键 词 的 出 现 频 次 ， 故 意 在 网 页 中 放 一 段 与 背景 颜色 相同 的 、 包 含 密集 关键 
字 的 文本 。 这 样 ， 访 问 网 页 的 用 户 看 不 到 ， 搜 索引 擎 却 能 找到 。 类 似 方法 还 包括 超 小 号 文 
字 、 文 字 隐 藏 层 等 手段 。 隐 形 链接 是 在 隐形 文本 的 基础 上 在 其 他 页 面 添 加 指向 目标 优化 页 
的 行为 。 以 下 是 一 个 隐形 文本 作弊 的 例子 。 

<head> 

<title>RASP-NET 案例 教程 </title> 

<style type="text/css"> 

/* 将 前 景色 背景 色 都 设置 为 白色 以 达到 隐藏 目的 */ 

-hiddenText1 { background-color:White; color:White; } 
/* 将 字体 设置 为 非常 小 ， 从 而 达到 隐藏 目的 */ 

.hiddenText2{ font-size:1lpx;} 
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</style> 


</head> 
<body> 
<div> 


这 一 段 是 正文 。 正 文 结束 。 

<span class="hiddenText1"> 
RSP.NET C# jQuery Web Service SQL Server 编程 项 目 案例 教程 
</span> <br /> 


这 是 <span 
另外 <span 
一 段 <span 
正文 <span 


。 正 文 结束 。 


</div> 
</body> 


5. 重 定向 


使 用 刷新 标记 、CGI 程序 、Javascript 或 其 他 技术 ， 当 用 户 进 入 该 页 时 ， 迅 速 自动 跳 转 
到 另 一 个 网 页 。 重 定向 使 搜索 引擎 与 用 户 访问 到 不 同 的 网 页 。 下 面 是 一 个 重 写 向 的 例子 ， 
这 个 页 面 中 包含 针对 搜索 引擎 进行 了 优化 的 页 面 ， 这 个 页 面 仅 被 搜索 引擎 搜索 到 ， 不 能 被 


class="hiddenText2">asp.net</span> 
class="hiddenText2">C#</span> 
class="hiddenText2"> 案 例 </span> 
class="hiddenText2"> 教 程 </span> 

BE /> 


用 户 看 到 ， 访 问 此 页 面 的 用 户 被 重 定向 到 另外 的 页 面 。 


<body> 


<!-- 下 面 是 隐藏 的 关键 字 --> 


<!-- 隐 藏 的 关键 字 --> 
<!-- 隐 藏 的 关键 字 --> 
<!-- 隐 藏 的 关键 字 --> 
<!- -隐藏 的 关键 字 --> 


<script type="text/javascript" language="javascript"> 


window.location = " 想 让 用 户 看 到 的 页 面 的 地 址 "; 
</script> 


这 是 页 面 正文 ， 是 给 搜索 引擎 看 的 ， 用 户 不 会 看 到 。 


</body> 


6. 复制 站 点 或 内 容 


通过 复制 整个 网 站 或 部 分 网 页 内 容 并 分 配 以 不 同 域名 和 服务 器 ， 以 此 欺骗 搜索 引擎 对 


同一 站 点 或 同一 页 面 进行 多 次 索引 的 行为 。 一 个 最 典型 例子 是 镜像 站 点 。 


7. 门 页 


门 页 也 称 桥 页 ， 针 对 某 一 关键 词 专门 制作 一 个 优化 的 页 面 ， 链 接 指向 或 重 定向 到 目标 
页 面 。 有 时 候 为 动态 页 面 建立 静态 入 口 ， 或 为 不 同 的 关键 词 建立 不 同 内 页 也 会 用 到 类 似 方 
法 ， 但 与 门 页 不 同 的 是 ， 前 者 是 网 站 实际 内 容 所 需 而 建立 的 ， 是 访问 者 所 需要 的 ， 而 门 页 


本 身 无 实际 内 容 ， 只 针对 搜索 引擎 作 了 一 堆 充斥 了 关键 词 的 链接 而 已 。 


URL 是 一 个 页 面 的 “名 片 ” 用 户 和 搜索 引擎 都 是 通过 URL 访问 页 面 。 一 个 简洁 、 易 
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懂 、 易 记 的 URL 能 够 使 页 面 更 加 友好 ， 也 能 够 提高 搜索 引擎 对 页 面 的 评价 。 本 节 将 介绍 
URL 设计 的 原则 ， 并 结合 ASPNET 编程 技术 说 明 如 何 通 过 重 写 URL 实现 优化 。 


6.2.1 静态 URL 和 动态 URL 


根据 URL 中 是 否 包含 检 索 字 符 串 〈 即 URL 中 问号 及 其 后 面 的 内 容 )， 可 以 将 URL 分 
成 静态 和 动态 两 种 类 型 。 静 态 URL 是 指 不 包含 检索 字符 串 的 URL, 静态 URL 通常 具有 更 
好 的 可 读 性 ， 也 更 加 容易 记忆 ， 下 面 是 两 个 静态 URL 的 例子 。 

http://www.mysite.com/home 

http://www.mycompany .com/help/aboutme.html 

动态 URL 是 指 包含 检索 字符 串 的 URL。 这 种 URL 在 ASPNET 开发 的 Web 项 目 中 很 
常见 ， 检 索 字 符 串 通常 用 来 给 页 面 传递 参数 。 以 下 是 两 个 动态 URL 的 例子 。 


http://www.mycompany.com/productdetail.aspx?pid=bookl 

http://www.mycompany.com/admin/userright .aspx?userid=admingroleid=01 

在 ASPNET 开发 的 项 目 中 ， 很 多 时 候 都 需要 使 用 同一 个 页 面 显示 不 同 的 内 容 ， 例 如 ， 
在 网 上 书店 项 目 中 ， 使 用 同一 个 页 面 BookDetail.aspx 显示 不 同 的 图 书信 息 。 为 了 给 
BookDetail.aspx 页 面 指定 要 显示 的 图 书 , 通常 需要 在 URL 中 添加 一 个 检索 字符 串 以 指定 图 
书 编 号 ， 这 样 就 形成 了 动态 URL。 

对 于 搜索 引擎 来 说 ， 静 态 URL 比 动态 URL 拥有 更 高 的 价值 。 在 使 用 ASPNET 开发 
Web 网 站 时 ， 考 虑 到 搜索 引擎 优化 的 因素 ， 应 该 尽量 避免 使 用 动态 URL。 那 么 ， 如 何 处 理 
检索 字符 串 和 静态 URL 之 间 的 矛盾 呢 ? URL 重 写 技术 很 好 地 解决 了 这 个 问题 。 


6.2.2 ”URL 重 写 概述 


URL 重 写 是 截取 传 入 的 Web 请 求 并 自动 将 请 求 重 定 向 到 其 他 资源 的 过 程 。 执 行 URL 
重 写 时 ， 通 常会 检查 被 请 求 的 URL， 并 基于 URL 的 值 将 请 求 重 定向 到 其 他 URL。 例 如 ， 
在 进行 网 站 重组 而 将 /people/ 目 录 下 的 所 有 网 页 移动 到 /info/employees/ 目 录 中 时 ， 需 要 使 用 
URL 重 写 来 检查 Web 请 求 是 否 指向 了 /people 目录 中 的 文件 。 如 果 请 求 指向 /people/ 目 录 中 
的 文件 ， 则 自动 将 请 求 重 定向 到 /info/employees/ 目 录 中 的 同一 文件 。 
利用 URL 重 写 技术 ,将 一 个 静态 URL 重 定向 到 一 个 动态 URL, 可 以 避免 让 用 户 和 搜 
索引 擎 看 到 动态 URL。 例 如 ， 当 用 户 请 求 BookDetail/12 页 面 时 ， 把 这 个 静态 URL 重 定向 
到 BookDetail.aspx?id=12 的 页 面 , 后 者 即 为 ASPNET 项 目 中 的 真实 页 面 , 可 以 通过 检索 字 
符 串 获得 页 面 参数 ， 并 根据 参数 动态 生成 页 面 内 容 。 而 这 一 切 对 用 户 来 说 都 是 透明 的 ， 用 
户 看 到 的 是 请 求 页 面 BookDetail12， 就 返回 了 编号 为 12 的 图 书信 息 ， 请 求 另外 一 个 页 面 
BookDetail/20， 就 返回 编号 为 20 的 图 书信 息 。 
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6.2.3 使 用 HTTP 模块 重 写 URL 


在 ASPNET 程序 中 , 处 理 每 一 个 HITP 请 求 时 都 会 执行 已 经 注册 的 HTTP 模块 , 所 以 
在 HTTP 模块 中 可 以 编写 适用 于 所 有 请 求 的 公用 代码 。 对 于 重 写 URL 来 说 , 在 得 到 一 个 请 
求 以 后 , 要 分 析 这 个 被 请 求 的 URL, 如 果 需 要 重 写 , 则 按照 重 写 规则 跳 转 到 重 写 后 的 页 面 。 
这 个 任务 适合 在 HTTP 模块 中 实现 。 下 面 通过 一 个 例子 说 明 具体 如 何 实现 。 

【 例 6-1】 简单 的 URL 重 写 。 

在 第 4 章 所 开发 的 网 上 书店 项 目 中 ， 有 许多 动态 URL, 一 个 典型 的 例子 是 图 书 详情 页 
面 。 当 用 户 在 图 书 列表 中 单 击 一 本 图 书 时 ， 就 会 转 到 图 书 详情 页 面 ， 并 将 所 选择 的 图 书 编 
号 作为 检索 字符 串 传递 给 图 书 详情 页 面 ， 形 成 类 似 于 BookDetail.aspx?id=10 的 动态 URL。 
为 了 将 动态 URL 优化 成 静态 URL， 希 望 使 用 BookDetail10 的 URL 访问 刚才 的 页 面 ， 这 
就 需要 URL 重 写 。 当 网 上 书店 程序 收 到 一 个 静态 URL 请 求 时 ， 将 其 重 写 为 动态 URL。 

(1) 打开 第 4 章 所 创建 的 网 上 书店 项 目 BookShop。 

(2) 在 项 目 中 添加 一 个 HTTP 模块 ， 命 名 为 MyUrlRewriter。 在 HttpApplication 的 
BeginRequest 事件 中 ， 检 查 所 请 求 的 URL 并 在 必要 时 进行 重 写 。 代 码 如 下 : 

public class MyUrlRewriter : IHttpModule 

1 


public void Dispose() 
{ } 
public void Init(HttpApplication context) 
i 
context .BeginRequest += new EventHandler (context BeginRequest); 
} 
void context BeginRequest (object sender, EventArgs e) 
{ 
HttpContext context = HttpContext.Current; 
string path = context .Request.Path.ToLower (); // 得 到 请 求 路 径 
if (path.StartsWith("/bookdetail/")) 
{ 
// 静 态 URL 形式 为 /bookdetail/6， 其 中 6 为 图 书 编号 ， 重 定向 动态 URL 
string id = path.Substring (path.LastIndexOf ('/') + 1); 
context .RewritePath ("~/BookDetail.aspx?id=" + id, false); 


} 


(3) 打开 图 书 列表 控件 BookListControl.ascx， 其 中 的 链接 地 址 原 为 动态 URL， 如 下 
代码 ; 

<a href="BookDetail.aspx?id=<%#Eval ("BookID")®%>"> 

<img src="BookImages/<%#Eval ("SmallImage") $>" alt="<%#Eval ("BookTitle") 

$>" style ="width: 100px;" /> 

</a> 
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将 上 述 代码 中 的 动态 URL 改 为 静态 URL， 代 码 如 下 : 


<a href="BookDetail/<%®#Eval ("BookID")%$>"> 

<img src="BookImages/<%#Eval ("SmallImage") %$>" alt="<%#Eval ("BookTitle") 
%>" style ="width: 100px;" /> 

</a> 


(4) 运行 网 上 书店 项 目 ， 从 图 书 列表 中 选择 一 个 图 书 查 看 详情 ， 可 以 看 到 图 书 详情 页 
面 的 URL 已 经 变 成 了 静态 URL。 运 行 界面 如 图 6.1 所 示 。 
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者 网 上 书店 大 学 优惠 芽 看 购物 于 
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设计 模式 ， 可 复 用 面向 对 象 软件 的 基础 


设计 模式 ， 可 复 用 面向 对 象 软件 的 基础 


会 员 价 格 ， 站 28. 00 

作者 : Erich gamma Richard Helm John 
出 版 社 ， 电 子 工业 出 版 社 

出 版 日 期 ，1938-1-1 


图 6.1 图 书 详情 URL 重 写 示 例 1 


(5) 从 图 6.1 中 可 以 看 到 ， 页 面 上 所 有 图 片 ( 包 括 项 部 横幅 、 图 书 封面 等 ) 都 不 能 显 
示 了 。 这 是 由 于 在 页 面 代 码 中 使 用 相对 路 径 引 用 图 片 ， 当 URL 重 写 后 路 径 就 发 生 了 变化 。 
例如 ， 图 书 封面 的 相对 路 径 为 BookImages/1b.jpg， 在 未 进行 URL 重 写 以 前 ， 这 个 图 片 的 
绝对 路 径 为 /BookImages/lbjpg， 而 进行 了 URL 重 写 后 ， 图 片 绝对 路 径 变 为 
/BookDetail/BookImages/1b.jpg。 为 了 解决 这 个 问题 ， 所 有 图 片 应 该 使 用 绝对 路 径 而 非 相 对 
路 径 。 在 母 版 中 修改 LOGO 和 横幅 图 片 路 径 ， 如 下 代码 所 示 〈 注 意图 片 路 径 是 以 /开头 ， 
表示 绝对 路 径 ， 如 果 没 有 这 个 /， 则 为 相对 路 径 )。 

<img src="/images/1logo.jpeg" alt=" 攀 登 者 网 上 书店 "” style="height:80px;" /> 

<img alt=" 大 学 优惠 "src="/images/banner.jpg" style="height:80px;" /> 


同样 的 道理 ， 在 BookDetail.aspx 页 面 中 需要 修改 图 书 封面 为 绝对 路 径 。 

Cover. ImageUT1 = "/BookImages/" + book.bigImage; 

(6) 如 果 页 面 上 的 JavaScript 和 CSS 使 用 相对 路 径 引用 , 则 也 存在 与 图 片 相同 的 问题 ， 
将 所 有 JavaScript 和 CSS 文件 引用 都 修改 为 绝对 路 径 。 


(7) 再 次 运行 项 目 ， 并 查看 某 图 书 详情 ， 可 以 看 到 所 有 图 片 都 能 正常 显示 ， 所 有 CSS 
样式 也 正常 应 用 ， 如 图 6.2 所 示 。 
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图 6.2 图 书 详情 URL 重 写 示 例 2 


6.2.4 ”处理 回 发 


URL 重 写 后 ， 由 于 改变 了 页 面 的 路 径 ， 会 带 来 一 系列 的 问题 ， 例 如 找 不 到 图 片 、 外 部 
CSS 文件 、 外 部 JavaScript 文件 等 ， eit tt ee 
如 例 6-1 中 所 采取 的 方案 一 样 。 除 此 以 外 ,由 于 URL 重 写 造成 页 面 路 径 改变 还 会 带 来 一 
更 加 严重 的 问题 ， 就 是 页 面 上 的 所 有 服务 器 端 事件 (如 按钮 的 服务 器 端 Click 事件 ) 各 
法 正常 工作 。 在 例 6-1 中 ， 在 图 书 详情 页 面 上 单 击 “ 加 入 购物 车 ”按钮 ， 则 会 出 现 如 图 6.3 
所 示 的 错误 界面 。 


无 法 找到 | 资源。 ~ Wozille Firefox 
文件 @) 编辑 到) 查看 WD 历史 GG) 书签 @) 工具 WD 帮助 0 


oe < /es Deomenl priel 77 ~ 


Googke 避 基 加- 号 -M- 回 :全 - 到 - 习 -4 S-®@- 
无 法 找到 光源 - -|| 


/应 用 程序 中 的 服务 器 错误 。 
无 涉 关 到 禾 。 


前 明 : HTTP 404。 和 正在 查 栈 的 次 亩 [或 者 它 的 一 个 估 电 ) 可 能 已 撤 黎 除 ， 或 其 名 称 已 更 改 ， 或 稍 时 不 可 用 。 访 查 以 下 URL 并 确保 其 本 二 正确 


请 家 的 URL: BookDetalGookDetal aspx 


图 6.3 单 击 按钮 后 产生 错误 


在 图 6.3 中 ， 提 示 无 法 找到 资源 /BookDetail/BookDetail.aspx。 从 浏览 器 的 地 址 栏 中 可 
以 看 到 ， 在 查看 图 书 详情 时 ， 页 面 URL 为 /BookDetai/l1 (其 中 1 为 正在 查看 的 图 书 编号 )， 
当 单 击 “ 加 入 购物 车 ”按钮 后 ， 页 面 地 址 变 为 /BookDetail/BookDetail.aspx?id=1。 这 种 情况 
带 来 两 个 问题 : 首先 是 这 个 页 面 地 址 不 存在 ， 造 成 无 法 找到 资源 ;其 次 是 这 个 地 址 又 变 成 
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了 URL 优化 以 前 的 动态 URL，URL 优化 失去 了 作用 。 

之 所 以 会 再 现 上 述 情况 , 是 由 于 URL 重 写 后 , 浏览 器 端 所 显示 的 路 径 与 服务 器 端 真实 
路 径 不 同 .具体 来 说 ,ASPNET 的 Web 窗 体 会 在 所 生成 的 HIML 代码 中 自动 添加 一 个 action 
属性 ， 这 个 属性 指明 了 页 面 所 要 提交 的 地 址 。ASPNET Web 窗 体 默认 为 自 提交 ， 即 提交 到 
页 面 本 身 。 在 例 6-1 中 ， 由 于 URL 重 写 ， 浏 览 器 端 比 服务 器 端 多 了 一 级 BookDetail 路 径 。 
图 书 详情 页 面 在 提交 时 ， 仍 然 提交 到 自身 ， 即 BookDetailaspx， 这 个 页 面 与 浏览 器 端 路 径 
相 结 合 ， 就 得 到 了 一 个 错误 的 路 径 /BookDetail/BookDetail.aspx， 于 是 出 现 了 图 6.3 所 示 的 

性 误 。 

URL 重 写 后 ， 不 仅 单 击 按钮 会 造成 如 图 6.3 所 示 的 错误 ， 其 他 服务 器 端 事件 也 会 出 现 
同样 错误 。 这 是 因为 ASPNET 服务 器 端 事件 都 是 提交 到 服务 器 端 页 面 本 身 , 然后 再 用 页 台 
代码 进行 处 理 。 由 于 URL 重 写 后 页 面 提 交 的 地 址 不 对 ， 找 不 到 要 提交 的 页 面 ， 就 会 出 现 如 
图 6.3 所 示 的 错误 。 

由 于 URL 的 重 写 造 成 页 面 无 法 提交 ,从 而 导致 服务 器 端 任何 事件 都 不 能 工作 , 这 是 一 
个 很 严重 的 问题 。 解 决 这 个 问题 的 方法 是 使 页 面 提交 时 仍然 提交 到 经 过 重 写 优化 以 后 的 
URL。 由 于 页 面 提交 的 地 址 是 在 页 面 生成 的 HTML 文档 中 form 元 素 的 action 属性 中 指定 ， 
要 想 修改 提交 地 址 ， 必 须 想 办 法 修改 页 面 所 生成 的 HTML 代码 中 的 action 属性 。 下 面 通过 
一 个 例子 来 说 明 具 体 实 现 步 又。 

【 例 6-2】 正确 处 理 回 发 。 

本 例 演示 了 如 何 解 决 URL 重 写 带 来 的 页 面 回 发 错误 ， 基 本 思想 是 重 写 form 元 素 的 
action 属性 ， 将 其 修改 为 经 过 优化 以 后 的 URL 〈 即 浏览 器 端 看 到 的 URL)。 这 个 解决 方案 
来 源 于 Scott Guthrie 的 博客 ， 其 中 用 到 了 ASPNET 控件 适配器 ， 读 者 可 以 参考 原文 以 获得 
更 多 信息 : http://weblogs.asp.net/scottgu/archive/2007/02/26/tip-trick-url-rewriting-with-asp- 
Det.aspX。 

(1) 打开 网 上 书店 项 目 。 

(2) 在 项 目 中 添加 一 个 控件 适配器 类 UrlRewriterControlAdapter 和 一 个 UrlRewriter 类 ， 
这 两 个 类 协作 完成 HtmlForm 类 的 自 定义 呈现 ， 修 改 了 最 终生 成 的 HTML 代码 中 的 action 
属性 。 代 码 如 下 : 

public class UrlRewriterControlAdapter:ControlAdapter 

{ 


protected override void Render (HtmlTextWriter writer) 


// 使 用 自 定义 的 UrlRewriter 类 呈现 控件 (此 处 为 HtmlForm) 


base.Render (new UrlRewriter (writer)); 


} 
public class UrlRewriter : HtmlTextWriter 
{ 

// 构 造 函 数 


public UrlRewriter (HtmlTextWriter writer) 
: base (writer) 

{ 
this.InnerWriter = writer.InnerWriter; 


} 


/// <summary> 
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/// 重 写 基 类 的 WriteRttribute () 方 法 
/// </summary> 
/// <param name="name"> 要 写 的 ARttribute 的 名 称 </param> 
/// <param name="value"> 要 写 入 的 Attribute 的 值 </param> 
/// <param name="fEncode"> 是 否 对 属性 名 称 和 值 进行 编码 </param> 
public override void WriteAttribute(string name, string value,bool 
fEncode) 
{ 
// 检 查 当前 要 写 入 的 属性 是 否 为 action 属性 


if (name.ToLower() == "action") 
{ 
var context = HttpContext.Current; 
if (context.Items["ActionRewrite"] == null) 


value = context.Request.RawUrl; 


// 将 其 重 写 为 浏览 器 请 求 的 Url 


context.Items["ActionRewrite"] = true; 


// 设 置 标记 ， 已 经 重 写 action 属性 
} 


base.WriteAttribute (name, value, fEncode); 


// 调 用 基 类 相应 方法 
! 


(3) 在 解决 方案 资源 管理 器 中 , 在 项 目 名 称 上 右 击 ， 从 弹出 的 快捷 菜单 中 选择 “添加 ” 
| “添加 ASPNET 文件 夹 ”|“App_Browsers” 命 令 ， 添 加 一 个 App_Browsers 文件 夹 。 
(4) 在 App_Browsers 文件 夹 中 ,添加 一 个 浏览 器 文件 my.browser， 并 在 其 中 注册 刚才 
所 创建 的 控件 适配器 类 UrlRewriterControlAdapter， 代 码 如 下 : 
<browsers> 
<browser refID="Default"> 
<controlAdapters> 
<adapter controlType="System.Web.UI.HtmlControls.HtmlForm" 
adapterType="BookShop.UrlRewriterControlAdapter" /> 
</controlAdapters> 


</browser> 
</browsers> 


(5) 运行 网 上 书店 ， 转 到 图 书 详情 页 面 ， 单 击 “ 加 入 购物 车 ”按钮 ， 可 以 看 到 ， 按 钮 
能 够 正常 工作 ， 再 转 到 查看 购物 车 页 面 ， 可 以 看 到 商品 已 经 被 放 入 购物 车 中 ， 从 而 说 明 这 
种 方案 成 功 解决 了 URL 重 写 后 的 页 面 回 发 问题 。 


6.3 ”正则 表达 式 与 URL 重 写 


在 6.2 的 例子 中 , 只 实现 了 一 个 页 面 即 BookDetail.aspx 的 URL 重 写 , 实现 的 代码 简单 
直观 ， 判 断 所 请 求 的 URL 是 否 具 有 类 似 “/bookdetail/1” 的 形式 ， 如 果 是 ， 则 将 URL 重 写 
为 动态 URL 的 形式 BookDetail.aspx?id=1， 代 码 如 下 : 


HttpContext context = HttpContext.Current; 
string path = context.Request.Path.ToLower (); 


// 得 到 请 求 路 径 
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if (path.StartsWith("/bookdetail/")) 

ii 
// 静 态 URL 形式 为 /bookdetail/1， 其 中 1 为 图 书 编号 ， 重 定向 动态 URL 
string id = path.Substring(path.LastIndexOof ('/') + 1); 
context .RewritePath ("~/BookDetail.aspx?id=" + id, false); 

} 


在 较 大 型 的 网 站 和 项 目 中 有 很 多 页 面 需 要 重 写 URL， 如 果 都 采用 上 述 方法 ， 则 每 个 页 
面 都 需要 写 一 个 if 或 者 switch 分 支 ， 代 码 很 复杂 。 而 且 ， 每 当 添 加 一 个 新 的 需要 重 写 的 
URL， 都 需要 修改 代码 ， 增 加 一 个 让 或 者 switch 分 支 。 这 种 URL 重 写 方式 不 灵活 ， 经 常 
需要 修改 ， 不 是 一 种 好 的 设计 。 使 用 正则 表达 式 可 以 实现 更 加 通用 的 解决 方案 。 


6.3.1 正则 表达 式 语法 


正则 表达 式 是 指 一 个 用 来 描述 或 者 匹配 ， 一 系列 符合 某 个 句法 规则 的 字符 串 的 单个 字 
符 串 。 一 个 正则 表达 式 通常 被 称 为 一 个 模式 〈patterm)， 被 广泛 用 于 各 种 字符 串 处 理 中 ， 如 
查找 、 替 换 、 格 式 验证 等 。 正 则 表达 式 定 义 了 一 个 字符 串 模式 ， 例 如 ， 规 定 了 字符 串 中 应 
该 包含 哪些 字符 ， 不 能 包含 哪些 字符 ， 以 什么 字符 开头 和 结尾 ， 某 些 字符 出 现 多 少 次 等 。 
正则 表达 式 本 身 也 是 一 个 字符 串 ， 用 一 种 特定 语法 描述 模式 ， 其 是 区 分 大 小 写 的 。 组 成 正 
则 表达 式 的 元 素 大 致 有 以 下 几 种 。 


1. 普通 字符 
普通 字符 表示 字符 本 身 。 例 如 ， 模 式 “abc” 表 示 包 含 abc 的 任意 字符 串 。 
2. 转 义 字符 


正则 表达 式 中 的 转 义 字符 与 C# 字 符 串 中 的 转 义 字 符 含义 类 似 。 转 义 字 符 可 以 看 成 是 普 
通 字符 的 一 种 特殊 形式 。 例 如 ， 模式“avtb” 匹 配 任意 一 个 字符 a 后 面 紧 跟 一 个 水 平 制 表 符 
再 紧 跟 一 个 字符 b。 

3. 元 字符 

元 字符 可 以 匹配 一 类 字符 ， 例 如 匹配 数字 、 字 符 、 空 白 等 。 常 用 的 元 字符 及 其 含义 
如 下 。 

口 [..]: 匹配 出 现 于 吕 中 的 任意 单个 字符 。 

口 [^...]: 匹配 不 在 [] 出 现 的 任意 单个 字符 。 

口 .: 匹配 除 换行 符 以 外 的 任意 单个 字符 。 

口 \w: 匹配 任意 单个 字母 、 数 字 、 下 划 线 、 汉 字 。 

口 \W: 匹配 任意 单个 不 属于 \w 模式 的 字符 。 

口 \s: 匹配 任意 单个 空白 字符 。 

口 \S: 匹配 任意 单个 非 空 白字 符 。 

口 \d: 匹配 单个 数字 字符 ， 相 当 于 [0-9]。 

口 \D: 匹配 任意 单个 非 数 字 字 符 ， 相 当 于 [^0-9]。 
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和 提示 : 如 果 要 在 字符 事 中 匹配 小 数 点 “”， 则 需要 在 模式 中 写成 人 ”的 形式 ， 即 在 “.” 
的 前 面 添 加 “\” 将 其 转 义 成 普通 字符 。 如 果 直 接 使 用 “.” 进 行 匹 配 ， 那 么 可 以 
匹配 除 换行 符 以 外 的 任意 单个 字符 。 


4. 开头 和 结尾 


在 定义 正则 表达 式 的 模式 时 ， 有 两 个 特殊 字符 ^ 和 $， 其 中 ^ 表 示 匹 配 字符 串 开头 ，$ 表 
示 匹 配 字符 串 结尾 。 例 如 ， 模 式 ^[A-Z] 匹 配 以 大 写 英文 字符 开头 的 字符 串 ， 模 式 \.$ 匹 配 以 
英文 句号 小数点) 结尾 的 字符 串 。 


和 提示 : 在 正则 表达 式 中 ,， “字符 有 两 种 完全 不 同 的 含义 。 当 ^ 出 现 于 方 括号 [] 内 时 ， 表 示 
“ 非 ”， 匹 配 不 出 现在 方 括号 [] 中 的 任意 字符 。 当 ^ 出 现在 方 括号 [] 外 时 ， 表 示 字 
符 串 开头 。 


5. 重复 


在 正则 表达 式 中 还 可 以 指定 元 素 出 现 的 次 数 ， 有 以 下 几 种 方式 。 

口 {mn}: 匹配 m 到 n 次 重复 ( 闭 区 间 ) 。 例 如 af1.3} 匹 配 1 到 3 个 字符 a。 

口 {m,}: 匹配 大 于 等 于 m 次 重复 。 例 如 a{2,} 匹 配 2 个 以 下 字符 a。 

口 {fm}: 匹配 正好 m 次 重复 。 

口 ?: 匹配 0 到 1 次 重复 ， 相 当 于 {0,1}。 这 种 重复 也 称 为 可 选 ， 即 可 以 出 现 也 可 不 
出 现 。 
口 +: 匹配 1 次 以 上 重复 ， 相 当 于 {1,}。 
口 *: 匹配 0 次 以 上 重复 ， 相 当 于 {0,}。 


6. 选择 
正则 表达 式 使 用 符号 | 表示 “或 ”的 关系 。 例 如 alA 表示 小 写字 母 或 者 大 写字 母 a。 
7. 分 组 和 引用 


正则 表达 式 使 用 圆 括号 〈) 将 多 个 字符 分 成 一 组 成 为 逻辑 上 的 一 个 单位 。 例 如 ， 模 式 
(\+0)?123 表示 括号 里 面 的 +0 是 可 选 的 (其 中 + 前 面 的 \ 为 转 义 符 ), 而 \+0?123 则 指 0 是 可 选 
的 。 字 符 串 +0123 和 123 都 能 够 匹配 这 两 个 模式 ， 字 符 串 +123 能 够 匹配 第 2 个 模式 但 不 能 
匹配 第 1 个 模式 。 

在 正则 表达 式 中 还 经 常 需要 引用 前 面 的 内 容 ， 例 如 ， 模 式 \d{3} 可 以 表示 一 个 三 位 数 ， 
那么 如 果 要 表示 这 个 三 位 数 重复 2 次 应 该 如 何 写 呢 ? 显然 不 能 用 \d{3}\d{3}, 这 个 模式 其 实 
表示 6 个 数字 ， 或 者 说 任意 一 个 6 位 数 。 如 果 要 表示 一 个 3 位 数 重 复 2 次 ， 那 么 就 需要 在 
第 1 个 3 位 数 后 面 再 出 现 与 前 一 个 3 位 数 完全 相同 的 内 容 ， 也 就 是 说 ， 在 模式 中 需要 引用 
前 面 出 现 的 内 容 。 
正则 表达 式 提供 的 分 组 和 引用 机 制 可 以 给 模式 中 出 现 的 一 部 分 内 容 分 组 ， 并 且 在 后 面 
的 模式 中 引用 这 个 分 组 。 如 前 所 述 ， 正 则 表达 式 使 用 圆 括 号 〈) 进行 分 组 ， 还 可 以 为 分 组 
指定 一 个 名 称 ， 语 法 为 在 紧 跟 在 左 侧 圆 括号 后 面 写 上 一 个 问号 和 一 对 包括 在 尖 括 号 中 的 名 


se 
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字 。 这 种 分 组 称 为 命名 分 组 ， 在 模式 中 可 以 通过 分 组 名 称 引用 某 个 分 组 。 如 刚才 所 说 的 一 
个 3 位 数 重 复出 现 2 次 可 以 用 以 下 模式 来 描述 : 
(2<groupl>\d{3}) \k<groupl> 
正则 表达 式 中 的 分 组 也 可 以 是 匿名 分 组 。 如 果 在 分 组 的 左 侧 圆 括号 后 面 没 有 写 问号 、 
尖 括 号 和 组 名 ， 则 这 个 分 组 为 匿名 分 组 。 引 用 匿名 分 组 的 方法 是 使 用 分 组 序号 ， 整 个 模式 
中 第 1 个 分 组 的 序号 为 1， 第 2 个 分 组 的 序号 为 2， 依次 类 推 。 可 以 通过 '\1 的 方式 引用 第 1 
个 分 组 。 例 如 ， 刚 才 所 说 的 3 位 数 重 复出 现 2 次 的 例子 也 可 以 用 以 下 模式 来 描述 : 
(Xdaf3}) AI 


6.3.2 正则 表达 式 验 证 


正则 表达 式 有 很 多 用 途 ， 其 中 的 一 个 常用 功能 就 是 验证 ， 即 验证 某 些 文本 是 否 符合 某 
种 模式 。 在 .NET Framework 中 ， 使 用 System.TextRegularExpressions 命名 空间 中 的 Regex 
类 表示 一 个 正则 表达 式 ， 通 过 在 Regex 类 的 构造 函数 中 传递 一 个 字符 串 参 数 指定 正则 表达 
式 模式 ， 代 码 如 下 : 


Regex r = new Regex(@"\d{6}"); 


Regex 类 的 Match 方法 执行 匹配 并 返回 一 个 Match 类 型 的 结果 ，Match 类 的 Success 
属性 表示 匹配 是 否 成 功 , Match 类 的 Value 属性 表示 匹配 成 功 的 文本 ,Match 类 的 NextMatch 
方法 执行 下 一 次 匹配 并 返回 匹配 结果 。 下 面 是 一 个 简单 的 例子 ， 查 找 字 符 串 中 出 现 的 数字 
并 输出 。 


Regex r = new Regex(@"\d+"); // 创 建 正则 表达 式 
Match m = r.Match("123aa45bb7cc89"); // 执 行 第 一 次 匹配 
while (m.Success) // 当 匹配 成 功 时 循环 
{ 
Console.WriteLine (m.Value); // 输 出 当前 匹配 的 文本 
m=m.NextMatch (); // 执 行 下 一 次 匹配 
} 
Ws 输出 结果 为 : 
123 
45 
i 
89 x 


如 果 读 者 仔细 观察 上 面 的 例子 ， 就 会 发 现 一 个 问题 : 正则 表达 式 的 模式 为 \d+， 表 示 匹 
配 一 到 多 个 连续 数字 ， 如 果 目 标 字符 串 中 包含 多 个 连续 数字 ， 那 么 这 个 模式 是 只 匹配 第 1 
个 数字 还 是 匹配 全 部 数字 呢 ? 根据 \d+ 模 式 的 含义 , 匹配 1 个 或 者 2 个 或 者 更 多 直到 全 部 数 
字 都 是 正确 的 ， 但 是 在 实际 执行 时 必须 返回 一 个 匹配 结果 。 应 该 返回 最 小 的 匹配 结果 还 是 
最 大 的 匹配 结果 ， 这 就 是 所 谓 的 “ 贪 楚 ”问题 。 

在 正则 表达 式 的 重复 模式 中 ， 默 认 使 用 贪 禁 方 式 进行 匹配 ， 就 是 说 ， 要 匹配 尽 可 能 多 
的 次 数 。 例 如 在 前 面 的 例子 中 , \d+t 总 是 匹配 最 多 的 连续 数字 。 但 是 贪 禁 匹配 也 不 总 是 最 贪 
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焚 的 ， 也 要 考虑 到 后 面 其 他 内 容 的 匹配 ， 考 虑 下 面 的 例子 。 


Regex r = new Regex(@"\wtend$"); 
Match m = 工 -Match("begin to end"); 
if (m.Success) 
| 
Console .WriteLine (" 匹 配 成 功 ， 贪 禁 匹 配 并 不 是 无 限制 的 贪 禁 ") ; 
} 


else 


Console .WriteLine(" 贪 禁 匹 配 太 贪 禁 ， 把 @ 也 匹配 了 ， 导 臻 整个 正则 表达 式 匹 配 失败 ") ; 


在 上 面 这 段 代码 中 ， 第 一 行 定义 了 一 个 模式 用 来 以 end 结尾 文字 ， 模 式 中 的 \w+ 可 以 
匹配 1 到 多 个 字母 、 数 字 、 下 划 线 、 汉 字 ， 当 然 也 可 以 匹配 end。 如 果 贪 禁 匹 配 把 end 也 
匹配 到 \w+ 中 ， 那 么 代码 中 定义 的 \w+end$ 就 是 永远 不 可 满足 的 。 而 上 面 的 代码 运行 的 结果 
却 显 示 匹 配 成 功 ， 说 明 贪 禁 匹 配 在 最 大 限度 的 匹配 更 多 字符 时 ， 也 要 受 其 后 模式 的 限制 ， 
或 者 说 ， 贪 禁 匹 配 是 在 满足 整个 模式 匹配 成 功 的 基础 上 ， 去 匹配 尽 可 能 多 的 重复 次 数 。 

与 贪 禁 相 反 的 另外 一 种 模式 匹配 称 为 非 贪 禁 匹 配 ， 非 贪 禁 匹 配 总 是 尽 可 能 匹配 最 少 的 
重复 次 数 。 通 过 在 重复 次 数 后 面 添加 一 个 问号 指定 该 重复 使 用 非 贪 禁 匹 配 ， 如 下 例 所 示 。 


Regex r = new Regex(@"\d+?"); // 使 用 非 灸 禁 匹 配 
Match m = r.Match("123aa45bb7cc89"); // 执 行 第 一 次 匹配 
while (m.Success) // 当 匹配 成 功 时 循环 
下 
Console .WriteLine (m.Value) // 输 出 当前 匹配 的 文本 
m=m.NextMatch () ; // 执 行 下 一 次 匹配 


上 面 的 代码 运行 时 ， 会 输出 九 行 分 别 显 示 1 到 9， 说 明 每 次 \d+? 模 式 只 匹配 了 一 个 数 
字 ， 即 满足 模式 的 最 小 重复 次 数 。 与 贪 禁 匹配 类 似 ， 非 贪 禁 匹配 也 要 受 整体 模式 匹配 的 限 
制 ， 为 了 使 整个 模式 匹配 成 功 ， 非 贪 禁 模 式 在 必要 时 也 会 匹配 更 多 重复 次 数 。 

使 用 Regex 进行 字符 串 模式 匹配 时 ， 默 认 情 况 是 区 分 大 小 写 ， 如 下 代码 会 匹配 失败 。 


Regex = new Regex(@"end$"); // 此 模式 要 求 以 end 结尾 
Match m = r.Match("TheEnd"); // 待 检验 文本 以 End 结尾 
if (m.Success) 
{ 

Console .WriteLine ("匹配 成 功 "); 
. 


else 
Console.WriteLine ("匹配 失败 ") ; // 由 于 大 小 写 不 一 致 匹配 失败 

如 果 要 使 Regex 忽略 字符 大 小 写 ， 则 需要 指定 一 个 选项 。Regex 类 有 一 个 Options 属 
性 ， 这 是 一 个 RegexOptions 枚 举 ， 描 述 了 正则 表达 式 匹 配 时 的 一 些 选 项 ， 如 是 否 区 分 大 小 
写 、 是 否 从 右 向 左 查找 匹配 、 是 否 执 行 多 行 模式 等 。 如 果 要 忽略 字母 大 小 写 , 可 以 在 Regex 
构造 函数 中 传递 一 个 RegexOptions.IgnoreCase 值 ， 代 码 如 下 : 

Regex rl = new Regex ("end$", RegexOptions.IgnoreCase); 

下 面 通过 一 个 综合 性 的 例子 演示 如 何 通 过 Regex 验证 常用 的 字符 串 模式 。 

【 例 6-3】 正则 表达 式 验证 。 

本 例 演示 使 用 Regex 验证 常用 的 字符 串 模 式 。 
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(1) 创建 一 个 控制 台 应 用 程序 RegexSample。 
(2) 在 项 目 中 添加 一 个 类 RegexTest 用 以 测试 正则 表达 式 ， 类 的 代码 如 下 : 


// 测 试 Regex 的 类 


class RegexTest 


{ 


// 测 试 数据 类 

class RegexTestData 
public string pattern; // 正 则 表达 式 模式 
public string description; // 模 式 的 说 明 


} 


public void test() 


{ 


List<RegexTestData> list = new List<RegexTestData>(); 
// 中 国 邮箱 由 6 位 数字 组 成 
RegexTestData test = new RegexTestData () 
{ 
pattern = "^[0-9] {6}$", 
description = "中 国 邮政 编码 " 
] 
list.Add (test); 
// 中 国电 话 号 码 : 3 一 4 位 区 号 + 横 线 - + 7 一 8 位 市 话 号 码 
test = new RegexTestData() 
{ 
pattern = @"^(\d{3}-I\d{4}-)?\d{8}1\d{7}$", 
description = "中 国电 话 号 码 " 
}; 
list.Add (test) 7 
//URL 地 址 : 多 个 字母 开头 ， 后 跟 ://， 然后 是 多 个 非 空白 字符 
test = new RegexTestData() 
{ 
pattern=@" [a-zA-z]+://[^\s] *", 
description="URL 地 址 " 
}; 
list.Add (test) 7 
/* 电子 邮箱 地 址 模式 分 析 : 
* 1 到 多 个 字母 数字 下 划 线 开头 ， 中 间 可 以 有 -+. 符 号 ， 必须 有 @ 符 号 
* @ 号 后 面 的 内 容 被 . 分 成 两 部 分 ， 每 一 部 分 的 模式 与 @ 号 前 面 的 模式 基本 相同 */ 


test = new RegexTestData() 


Pattezn=e \wt([=+s]\wt)*eN\wr (LI\wt) Ne Nw (LN 
description=" 电 子 邮 件 地 址 " 

}; 

list.Add (test); 

For (Ine Le= 0 3 < Dist COE EL) 
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Console -Write (" 请 输入 一 个 10}:",1ist[i]l .description) 
string s = Console.ReadLine(); 
testMatch(s, list[il].pattern); 


} 
/// <summary> 
/// 测试 正则 表达 式 是 否 匹配 
/// </summary> 
/// <param name="text"> 要 测试 的 文本 </param> 
/// <param name="pattern"> 用 来 测试 的 模式 </param> 
private void testMatch (string text,string pattern) 
Regex r = new Regex(pattern); 
Match m = r.Match (text); 
if (m.Success) 
Console.WriteLine ("匹配 成 功 ， 结 果 为 :"+m.Value); 
// 匹 配 成 功 则 输出 匹配 结果 
else 
Console.WriteLine ("匹配 失败 "); 


(3) 在 Main0 方 法 中 调用 以 上 测试 方法 。 


static void Main(string[] args) 


出 


new RegexTest () .test (); 
Console.ReadKey (); 


(4) 运行 此 程序 ， 运 行 结果 如 下 : 

请 输入 一 个 中 国 邮 政 编码 :100123 

匹配 成 功 ， 结 果 为 :100123 

请 输入 一 个 中 国电 话 号 码 :021-12345678 
匹配 成 功 ， 结 果 为 :021-12345678 

请 输入 一 个 URL 地 址 :http://test.com 
匹配 成 功 ， 结 果 为 :http://test.com 

请 输入 一 个 电子 邮件 地 址 :my-this@163. com 
匹配 成 功 ， 结 果 为 :my-this@163.com 


6.3.3 正则 表达 式 查找 和 蔡 换 


1 


正则 表达 式 可 以 实现 复杂 的 字符 串 查 找 和 替换 。 在 .NET Framework 中 ,使 用 正则 表达 


式 进行 字符 串 查 找 时 , 除了 Regex 类 以 外 , 还 需要 用 到 Capture、Group 和 Match 类 。Capture 
类 表示 一 个 捕获 结果 有 3 个 重要 属性 : Index 表示 捕获 的 字符 串 在 整个 字符 串 中 的 位 置 ， 
Length 表示 所 捕获 字符 串 的 长 度 ，Value 表示 所 捕获 字符 串 的 值 。Group 类 表示 模式 匹配 中 
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的 一 个 组 ，Group 类 从 Capture 类 继承 。Match 类 表示 一 次 匹配 ，Match 类 从 Group 类 继承 。 
Group 类 有 一 个 CaptureCollection 类 型 的 Captures 属性 , 表示 这 一 组 所 有 捕获 的 集合 .Match 
类 有 一 个 GroupCollection 类 型 的 Groups 属性 ， 表 示 这 个 匹配 的 所 有 组 的 集合 。 这 几 个 类 


之 间 的 关系 如 图 6.4 所 示 。 
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图 6.4 正则 表达 式 类 关系 图 


Capture、Group、Match 这 几 个 类 之 间 的 关系 较为 复杂 ， 既 有 继承 关系 又 有 集合 关系 。 
通过 正确 理解 并 合理 使 用 这 些 类 ， 可 以 完成 复杂 的 字符 串 操作 。 下 面 通 过 两 个 例子 来 演示 


这 几 个 类 的 关系 及 应 用 。 
【 例 6-4】 捕获 、 组 和 匹配 。 


本 例 演示 捕获 Capture、 组 Group 和 匹配 Match 类 的 应 用 及 其 关系 。 
(1) 创建 一 个 控制 台 应 用 程序 ， 在 Main0 方 法 中 编写 以 下 代码 。 


static void Main() 


int count; 
Regex regex = new Regex("(123)+"); 
string text = "abc123123123abc123123abc"7 
Match match = regex.Match (text); 
while (match.Success) 
Console.WriteLine ("找到 一 个 新 的 匹配 ") ; 


GroupCollection groups = match.Groups; 


// 匹 配 一 到 多 个 123 
// 要 查找 的 字符 串 
// 执 行 一 次 匹配 

// 若 匹配 成 功 则 循环 


// 获 得 此 次 匹配 组 的 集合 


Console.WriteLine (" 该 匹配 共有 {0} 组 ",groups-Count) ; 


for (int 1 = .02 1 < groups:.Count: 1L++) 
{ 
Group g = groups[i]; 
Console.WriteLine ("\t 新 的 一 组 "); 


CaptureCollection captures = g.Captures; 
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// 获 得 当前 组 的 所 有 捕获 
Count = captures.Count; 


Console-WriteLine ("\t 该 组 中 共有 {0} 个 捕获 "，count); 
for (int 07 了 < count; j++) 
{ 


Capture c = captures[j]; 
// 显 示 一 个 捕获 的 详细 信息 
Console .WriteLine("\t\t 捕获 详情 : 位 置 : {0}, 文本 : {1}",c.Index, 
c.Value); 
二 
match = match.NextMatch () // 尝 试 下 一 个 匹配 
上 


(2) 运行 上 述 程序 ， 输 出 结果 如 下 : 


找到 一 个 新 的 匹配 
该 匹配 共有 2 组 
新 的 一 组 
该 组 中 共有 1 个 捕获 
捕获 详情 : 位 置 :3， 文 本 :123123123 
新 的 一 组 
该 组 中 共有 3 个 捕获 
捕获 详情 : 位 置 :3， 文 本 :123 
捕获 详情 : 位 置 :6， 文 本 :123 
捕获 详情 : 位 置 :9， 文 本 :123 
找到 一 个 新 的 匹配 
该 匹配 共有 2 组 
新 的 一 组 
该 组 中 共有 1 个 捕获 
捕获 详情 : 位 置 :15， 文 本 :123123 
新 的 一 组 
该 组 中 共有 2 个 捕获 
捕获 详情 : 位 置 :15， 文 本 :123 
捕获 详情 : 位 置 :18， 文 本 :123 


【 例 6-5】 HTML 文本 提取 。 

本 例 演示 如 何 从 HTML table 中 去 除 <tr><td> 等 标记 ， 读 取 其 中 的 文字 。 

问题 分 析 : 根据 对 HTML 中 table 的 格式 可 知 ，table 中 能 够 显示 在 浏览 器 中 的 内 容 就 
是 <td> 标 记 中 的 内 容 。 因 此 ， 要 提取 table 中 的 内 容 ， 只 需 取 出 <tdt> 和 </td> 标 记 之 间 的 文本 
即 可 。 用 正则 表达 式 很 容易 实现 这 个 功能 。 

(1) 创建 一 个 控制 台 应 用 程序 ， 并 编写 如 下 代码 。 


static void getHtml () // 版 本 1 
{ 
// 原 始 HTML 字符 串 
string table = "<table>" 
+"<tr><td>one</td><td>two</td></tr>" 
+"<tr><td>three</td><td>four</td></tr>" 
+"</table>", 
Regex regex = new Regex("<td>(.)+</td>"); 
// 正 则 表达 式 
Match match = regex.Match (table); 
while (match.Success) 
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} 
C 


Console.WriteLine (match.Value); 
match = match.NextMatch(); 


2) 运行 上 述 代 码 ， 但 是 并 没有 得 到 期 望 的 输出 ， 而 是 输出 以 下 内 容 : 


<td>one</td><td>two</td></tr><tr><td>three</td><td>four</td> 


观察 上 述 输 出 结果 ， 可 以 看 出 匹配 的 整个 字符 串 的 第 一 个 <td> 和 最 后 一 个 </td> 之 间 的 


内 容 ， 


这 是 由 于 贪 焚 匹 配 造成 的 。 正 则 表达 式 默认 使 用 贪 禁 匹 配 ， 总 是 匹配 尽 可 能 多 的 字 


符 ， 而 定义 的 模式 为 以 <td> 开 头 、 以 </td> 结 尾 的 字符 串 ， 按 照 贪 禁 匹 配 原则 ， 就 匹配 了 
最 开始 的 <td> 和 最 末尾 的 </td>。 


( 


3) 为 了 解决 贪 禁 匹配 带 来 的 问题 ,正确 取出 <td> 和 </td> 之 间 的 内 容 ， 需 要 使 用 非 贪 


禁 匹 配 ， 对 前 面 的 代码 做 以 下 修改 (只 给 出 修改 部 分 )。 


Regex regex 
Match match 


new Regex("<td>(?<g1>.+?)</td>"); // 使 用 非 贪 禁 匹 配 , 命名 分 组 
regex.Match (table); 


while (match.Success) 


Console.WriteLine (match.Groups["g1"] .Value);  // 输 出 命名 分 组 中 的 内 容 
match = match.NextMatch(); 


4) 再 次 运行 程序 ， 这 次 输出 了 正确 结果 如 下 : 


one 
two 
three 
four 


在 处 理 字符 串 时 ， 除 查找 功能 外 ， 也 经 常用 到 替换 功能 。 例如， 在 URL 搜索 引擎 优化 
中 ， 需 要 把 一 个 动态 URL 重 写 为 静态 URL， 从 动态 URL 到 静态 URL 的 转换 就 是 一 个 字 
符 串 替换 的 过 程 。Regex 类 提供 了 一 个 Replace() 方 法 可 以 实现 字符 串 替换 ,方法 签名 如 下 。 


public string Replace (string input, string replacement); 


Regex 类 Replace() 方 法 的 功能 在 指定 的 输入 字符 串 内 ， 使 用 指定 的 蔡 换 字符 串 替 换 与 
某 个 正则 表达 式 模式 匹配 的 所 有 字符 串 。 该 方法 第 一 个 参数 input 为 要 替换 的 原始 字符 串 ， 


replac 


ement 参数 为 奉 换 字符 串 ， 方 法 返回 替换 以 后 得 到 的 字符 串 。 例 如 ， 下 面 的 代码 把 字 


符 串 中 的 多 个 连续 空格 蔡 换 为 一 个 空格 。 


Regex regex = new Regex(@"\s+"); 
string text = "this is the original text ey 
string newText=regex.Replace (text, ™ "); 


在 字符 串 替换 时 ， 有 时 候 需 要 用 前 面 出 现 的 字符 串 来 替换 后 面 的 某 些 字符 串 ， 例 如 ， 
英语 中 姓名 有 两 种 写法 ， 名 在 前 或 者 姓 在 前 ， 当 姓 在 前 名 在 后 时 需要 在 姓 和 名 之 间 添 加 一 
个 加 号 。 例如， 比尔 盖 芯 的 名 字 可 以 写作 Bil Gates 或 者 Gates, Bill 都 可 以 。 如 果 要 把 一 个 
人 的 英文 名 字 从 名 在 前 的 格式 替换 为 姓 在 前 的 格式 ， 则 可 以 使 用 以 下 代码 实现 。 
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string input="Sun Ji-Lei"™; 

// 定 义 模式 (其 中 包含 两 个 命名 分 组 ) 

Regex regex=new Regex(@"(?<first>\S+) (?<last>\S+)"); 

// 执 行 替换 ， 使 用 {$ 组 名 } 方 式 引 用 原始 字符 串 中 的 分 组 

string output = regex.Replace (input,@"${1last},${first}") ;上述 代码 


6.3.4 正则 表达 式 URL 重 写 


使 用 正则 表达 式 强大 的 匹配 替换 功能 ,可 以 实现 更 加 灵活 通用 的 URL 重 写 。 其 基本 原 
理 也 是 使 用 HTTP 模块 捕获 所 有 访问 请 求 ， 将 请 求 的 URL (通常 是 静态 URL) 按照 一 定 的 
模式 进行 替换 ， 得 到 一 个 新 的 URL (通常 是 动态 URL)， 然 后 访问 这 个 重 写 后 的 URL。 
根据 以 上 理论 ， 如 果 要 自己 编码 实现 利用 正则 表达 式 重 写 URL， 需 要 在 例 6-1 的 基础 
上 ，, 在 HITP 模块 中 定义 重 写 URL 所 使 用 的 正则 表达 式 ， 并 使 用 这 些 正则 表达 式 对 静态 
URL 进行 替换 即 可 。 目 前 网 络 上 也 有 几 个 开源 的 URL 重 写 工具 ， 如 UrlRewriterNET (网 
址 http://urlrewriter.net/)、UrlRewritingNet( 网 址 http://www.urlrewriting.net/149/en/home.html) 
等 ， 都 支持 正则 表达 式 。 这 两 个 URL 重 写 工具 都 使 用 配置 文件 定义 重 写 的 正则 表达 式 。 下 
面 以 UrlRewriterNet 为 例 说 明 如 何 实现 URL 重 写 。 

【 例 6-6】 UrlRewriterNet 实现 URL 重 写 。 

假设 有 一 个 面向 全 球 各 个 国家 的 销售 网 站 ， 使 用 多 种 语言 向 客户 展示 商品 信息 。 网 站 
根 目 录 下 有 一 个 Detail.aspx 页 面 ， 用 来 显示 商品 信息 。 访 问 此 页 面 时 需要 指定 两 个 参数 : 
商品 编号 和 界面 语言 。 动 态 URL 使 用 检索 字符 串 传 递 参数 ， 具 有 类 似 这 种 形式 : 
Detail.aspx?language=en&id=10, 其 中 en 表示 语言 为 英语 , 10 表示 要 查看 的 商品 编号 为 10。 
为 了 实现 URL 优化 ， 希 望 用 户 使 用 静态 URL 访问 这 个 页 面 ， 静 态 URL 具有 以 下 形式 : 
/en/Detail10.aspx。 本 例 演 示 如 何 使 用 UrlRewriterNET 实现 这 个 功能 。 

(1) 在 ASPNET 项 目 中 添加 对 Intelligencia.UrlRewriter 程序 集 的 引用 。 

(2) 在 web.config 文件 的 configSection 节 中 添加 以 下 代码 ， 这 些 代码 的 作用 是 告诉 
UrlRewriterNET 从 web.config 文件 的 rewriter 结 点 中 读 取 配置 信息 。 


<configSections> 
<section name="rewriter" requirePermission="false" 
type="Intelligencia.UrlRewriter.Configuration. RewriterConfiguratio- 
nSectionHandler, Intelligencia.UrlRewriter" /> 
</configSections> 


(3) 在 web.config 文件 中 注册 UrlRewriterNET 所 使 用 的 HTTP 模块 。 


<httpModules> 

<add name="UrlRewriter" 
type="Intelligencia.UrlRewriter.RewriterHttpModule, Intelligencia. 
UrlRewriter"/> 

</httpModules> 


(4) 在 web.config 文件 中 用 添加 用 正则 表达 式 描 述 的 URL 重 写 规则 。 
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页 


的 


<rewriter> 


<rewrite url="~/(.*)/Detail (.*) .aspx" to="~/Detail.aspx? language=$1&amp; 


id=$2" /> 
</rewriter> 


(5) 为 了 测试 URL 重 写 效果 ， 在 项 目 中 添加 一 个 页 面 Detail.aspx。 出 于 演示 目的 ， 本 


语言 和 产品 编号 参数 。 


<form id="forml" runat="server"> 
<div> 


不 根据 参数 显示 不 同 的 语言 界面 和 产品 信息 , 而 只 是 在 页 面 上 显示 从 QueryString 获得 


你 所 使 用 的 语言 为 : <asp:Label Text="" runat="server" ID="language"/> <br /> 
你 所 要 查看 的 产品 为 : <asp:Label Text="" runat="server" ID="product" /><br /> 


</div> 
</form> 


(6) 在 Detail.aspx.cs 文件 中 ， 在 Page_Load 事件 中 显示 当前 的 语言 和 产品 编号 。 


protected void Page Load(object sender, EventArgs e) 


if (!IsPostBack) 
Product .Text = Request.QueryString["id"]; 


language .Text = Request.QueryString["language"]; 


(7) 在 网 站 中 添加 一 个 HTML 页 面 ， 页 面 中 放置 几 个 超 链接 ， 


<body> 

<a href="en/detail01.aspx">en, 产 品 01</a><br /> 

<a href="zh/detailp10.aspx"> 中 文 , 产 品 p10</a><br /> 
</body> 


(8) 运行 网 站 ， 运 行 结果 如 图 6.5 所 示 。 


文件 编辑 到 ) 查看 WW 历史 GE) 书签 @ 工具 0) 帮助 人 D 


以 打开 产品 详情 页 面 。 


党 


鲜于 © | tp://ocdhost:1092/webl/zh/ detailpl0. aspx/ -| 


Go| Ht OM OQ: 


ID http://localhos"etailplO. aspz/ | 了 


所 .@- 


你 所 使 用 的 语言 为 ， zh 
你 所 要 查看 的 产品 为 ，p10 


图 6.5 URL 重 写 效 果 


6.4 页 面 内 容 优化 


对 于 搜索 引擎 优化 而 言 ，URL 优化 仅 是 较 简单 的 一 步 ， 更 重要 的 还 是 要 对 页 面 内 容 进 


行 优化 。 页 面 内 容 优化 要 比 URL 优化 更 加 复杂 ， 本 节 将 介绍 页 面 内 容 优化 的 主要 原则 和 


方 


法 。 


05s 


第 1 篇 ASPNET 网 络 开发 关键 技术 


6.4.1 页 面 代码 优化 


页 面 代码 优化 是 指 通 过 合理 编写 页 面 代码 (包括 HTML、JavaScript 等 )， 达 到 突出 页 
面 主题 和 主要 内 容 ， 减 少 页 面 文 件 大 小 的 目的 。 页 面 代码 的 优化 主要 包括 以 下 儿 个 方面 。 


1. 合理 使 用 关键 字 


搜索 引擎 检索 页 面 信息 时 ， 根 据 页 面 内 容 来 判断 页 面 是 否 与 所 搜索 的 关键 字 匹配 ， 关 
键 字 在 页 面 上 出 现 的 次 数 越 多 ， 一 般 来 说 就 意味 着 页 面 与 被 搜索 关键 字 相 关 程 度 越 高 。 当 
然 ， 如 本 章 6.1 节 所 述 ， 关 键 字 的 出 现 必 须 是 合理 的 ， 如 果 堆 砌 关键 字 就 成 为 搜索 引擎 作 
浆 ， 会 受到 搜索 引擎 的 惩罚 。 
关键 字 应 该 在 页 面 代 码 的 <head> 部 分 和 <body> 部 分 都 出 现 。 在 <head> 部 分 ， 可 以 在 
<title> 中 为 页 面 指定 一 个 包含 关键 字 的 标题 ， 在 <meta> 中 指定 页 面 的 描述 信息 和 关键 词 。 
在 <body> 部 分 ， 即 页 面 显示 的 正文 中 ， 也 要 注意 让 关键 字 合理 出 现 。 例 如 ， 下 面 是 一 个 介 
绍 ASPNET 编程 案例 的 页 面 ， 较 好 地 使 用 关键 字 说 明了 页 面 的 主题 。 
<head runat="server"> 
<meta name="description" content=" 介 绍 了 ASP.NET 编程 知识 和 一 个 真实 项 目 案 
例 " /> 
<meta name="keywords" content="ASP.NET, 编程 , 案例 , 教程” /> 
<title>ASP.NET 项 目 案例 </title> 
</head> 
<body> 
<h1>ASP .NET 项 目 教程 之 案例 篇 </h1> 


ee 
2. 消除 元 余 代 码 


当 使 用 所 见 即 所 得 的 可 视 化 开发 工具 (如 Visual Studio、Dreamweaver 等 ) 编辑 页 面 
时 ， 会 产生 一 些 见 余 代 码 。 页 面 中 包含 这 些 代码 不 但 没有 任何 作用 ， 反 而 增加 页 面 大 小 ， 
扰乱 页 面 结构 。 例 如 当 使 用 Visual Studio 插入 和 编辑 表格 后 ， 会 产生 许多 空格 字符 和 样式 
信息 ， 如 下 代码 所 示 。 
<style type="text/css"> 
.stylel {width: 82px;} 


.style2{width: 95px;} 
.style3{width: 83px;} 


</style> 
<table> 
<Er> 
<td class="stylel"> 
学 号 </td> 
<td class="style2" > 
姓名 </td> 
<td class="style3"> 
年 龄 </td> 
<WER> 
<tr> 


<td class="stylel™" > 
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&nbsp;</td> 
<td class="style2"> 
&nbsp; </td> 
<td class="style3"> 
&nbsp; </td> 
/EES 
</table> 
上 述 代码 中 ,<td> 标 记 中 的 &nbsp: 是 多 余 的 ,从 第 2 个 <tr> 开 始 , 其 中 的 class="stylel" 
也 是 多 余 的 。 把 这 些 元 余 代 码 删除 以 后 ， 一 方面 减 小 了 页 面 字 节 数 ， 另 一 方面 使 页 面 更 加 


清晰 易 读 。 
3. 突出 重点 内 容 


无 论 对 于 搜索 引擎 还 是 用 户 而 言 , 页 面 中 的 某 些 内 容 显得 比 其 他 内 容 更 为 重要 , 例如 ， 
页 面 中 的 标题 〈 在 各 级 <h> 标 记 中 的 内 容 )、 页 面 上 用 粗 体 显示 的 字体 等 。 当 人 们 在 浏览 页 
面 时 ， 这 些 元 素 会 受到 更 多 重视 ， 搜 索引 擎 也 模拟 人 的 这 一 行为 ， 认 为 这 些 元 素 更 能 说 明 
页 面 内 容 。 根 据 这 一 点 ， 应 该 合理 使 用 标题 和 加 粗 字 体 ， 突 出 页 面 主 题 ， 而 不 能 整个 页 面 
都 使 用 一 样 的 文本 格式 。 

4. JavaScript 导 航 优化 

搜索 引擎 是 通过 页 面 之 间 的 链接 对 整个 互联 网 进行 搜索 的 。 如 果 一 个 页 面 没 有 任何 外 
部 链接 ， 那 么 这 个 页 面 不 可 能 被 搜索 引擎 蜘蛛 访问 到 ， 更 不 可 能 被 检索 收录 。 目 前 ， 搜 索 
引擎 只 识别 <a hre 伟 "somepage"> 标 记 为 链接 标记 ， 根 据 href 属性 访问 到 所 指向 的 页 面 。 在 
设计 网 站 时 ， 大 多 数 链接 都 是 通过 <a> 实 现 的 ， 但 是 也 存在 例外 情况 ， 最 常见 的 就 是 有 时 
候 需 要 用 JavaScript 打开 一 个 窗口 ， 如 下 代码 所 示 。 

<a href="#" onclick="window.open('mypage.aspx', 'mywin');"” >ASP.NET 项 目 案 

例 </a> 

对 于 搜索 引擎 的 蜘蛛 程序 来 说 ， 认 为 上 述 代 码 中 <a> 标 记 没有 目标 链接 页 面 ， 从 而 无 
法 访问 到 mypage.aspx。 可 是 如 果 直 接 把 页 面 地 址 写 到 href 属性 中 ， 又 无 法 实现 弹出 窗口 
的 效果 。 这 时 可 以 采用 一 个 小 技巧 ， 就 是 一 方面 在 href 属性 中 包含 要 链接 的 真实 地 址 ， 这 
样 搜索 引擎 就 能 访问 到 目标 页 面 , 别 一 方面 在 onclick 事件 中 用 window.open 打开 目标 窗口 ， 
然后 返回 false， 以 避免 打开 <a> 元 素 的 目标 地 址 ， 如 下 代码 所 示 。 


<a href="#" onclick="window.open('mypage.aspx', 'mywin');return false;" 
>ASP .NET 项 目 案 例 </a> 


6.4.2 消除 重复 内 容 


如 本 章 第 1 节 所 述 ， 如 果 2 个 或 多 个 页 面包 含 基本 相同 的 内 容 ， 会 降低 搜索 引擎 对 页 
面 的 评价 。 由 于 各 种 原因 , 有 些 重 复 页 面 是 不 可 避免 的 ,最 常见 的 情况 就 是 打印 机 友好 (print 
friendly) 页 面 。 

大 多 数 的 网 页 设计 是 为 了 在 计算 机 屏幕 上 观看 的 。 然 而 ， 有 的 时 候 用 户 需 要 将 某 些 页 
面 打 印 出 来 ， 也 许 就 是 为 了 保留 一 个 长 期 的 记录 ， 或 者 将 其 用 作 方 便 的 离线 参考 资料 。 但 
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随 之 而 来 的 问题 是 ， 显 示 器 是 彩色 的 ， 而 大 多 数 打印 机 都 是 黑白 的 ， 在 显示 器 上 看 起 来 引 
人 注目 和 五 彩 缤纷 的 很 多 特性 ， 都 无 法 在 打印 机 上 表现 出 相同 的 效果 。 在 被 降级 为 灰 度 打 
印 的 时 候 ， 彩 色 的 组 合 会 失去 原 有 的 对 比 效 果 ; 图 形 会 看 起 来 失真 ， 而 且 耗 费 太 长 的 打印 
时 间 ; 在 Web 页 面 上 起 着 重要 作用 的 导航 按钮 在 打印 页 面 上 也 毫 无 用 处 。 

为 了 克服 这 些 问题 ， 网 页 开发 人 员 常 常会 为 页 面 专 门 设计 一 个 打印 机 友好 的 版 本 。 打 
印 机 友好 的 版 本 通常 都 包括 有 和 主要 Web 页 面相 同 的 内 容 ， 但 是 会 省 略 掉 大 多 数 的 图 形 、 
背景 和 导航 元 素 。 页 面 还 会 把 彩色 转换 成 某 种 形式 ， 以 便 生 成 能 够 让 人 接受 的 灰 度 图 像 。 

打印 机 友好 页 面 很 好 的 解决 了 显示 和 打印 页 面 的 矛盾 ， 但 是 却 在 搜索 引擎 优化 方面 带 
来 了 另外 一 个 问题 ， 就 是 页 面 内 容重 复 。 如 前 所 述 ， 显 示 页 面 和 打印 机 友好 页 面 上 的 内 容 
几乎 是 相同 的 ， 只 是 显示 样式 和 图 片 不 同 。 从 搜索 引擎 优化 的 角度 来 说 ， 希 望 搜索 引擎 只 
能 看 到 其 中 的 一 个 页 面 ， 通 常 是 显示 页 面 。 为 了 达到 这 个 目的 ， 可 以 在 页 面 中 添加 一 个 
<meta> 标 记 以 拒绝 搜索 引擎 对 此 页 面 的 索引 ， 如 下 代码 所 示 。 


<meta name="robots" content="noindex,nofollow" /> 


上 述 代 码 中 的 robots 表示 这 个 内 容 是 针对 搜索 引擎 蜂 蛛 程序 的 ,noindex 表示 不 索引 当 
前 页 面 内 容 ，nofollow 表示 不 跟踪 当前 页 面 中 的 链接 。 


6.5 小 结 


作为 一 个 ASP.NET 开发 人 员 ， 需 要 掌握 基本 的 搜索 引擎 优化 知识 ， 使 开发 的 网 站 有 具 
有 较 好 的 搜索 引擎 评价 。 本 章 介绍 了 搜索 引擎 优化 的 基础 知识 ， 先 介绍 了 搜索 引擎 优化 的 
原理 和 误区 ， 然 后 重点 介绍 了 两 种 搜索 引擎 优化 手段 : URL 优化 和 页 面 内 容 优化 。 
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第 7 章 Visual Studio 2010 新 特性 


Visual Studio 2010 开发 环境 包含 了 新 的 .NET Framework 版 本 和 新 的 C# 语 言 版 本 ， 还 
添加 了 一 种 新 的 编程 语言 F#，Visual Studio 集成 开发 环境 也 做 了 改进 和 完善 ， 本 章 将 介绍 
Visual Studio 2010 在 开发 环境 和 语言 方面 的 新 特性 。 


7.1 集成 开发 环境 的 改进 


Visual Studio 2010 与 Visual Studio 2008 开发 环境 相 比 在 外 观 上 有 较 明显 的 差别 , 在 文 
本 编辑 、 调 试 等 各 方面 做 了 增加 和 改进 。 


7.1.1 新 的 窗口 风格 


Visual Studio 2010 使 用 WPF 开发 ,其 界面 风格 与 以 前 的 版 本 相 比 有 明显 的 差别 , Visual 
Studio 2010 旗舰 版 起 始 页 界面 如 图 7.1 所 示 。 


MG FD OM 


SI -上 av BE 


入 门 ， 指南 向 资源 最 新 基 闻 


图 7.1 Visual Studio 2010 起 始 页 


Visual Studio 2010 中 的 窗口 具有 更 强 的 依靠 功能 ， 各 个 窗口 (如 代码 窗口 、 设 计 窗 口 、 
调试 窗口 、 输 出 窗口 等 ) 不 但 可 以 停靠 在 Visual Studio 主 窗口 的 任何 一 侧 ， 而 且 还 可 以 脱 
离 Visual Studio 主 窗口 成 为 一 个 独立 的 窗口 ， 当 然 也 可 以 方便 地 再 附加 到 Visual Studio 主 
窗口 中 。 在 任意 一 个 窗口 的 标题 栏 双击 , 即 可 将 其 脱离 Visual Studio 主 窗口 成 为 独立 窗口 ， 
如 图 7.2 所 示 。 
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ET 
ek 


图 7.2 脱离 Visual Studio 主 窗口 的 独立 窗口 


7.1.2 ”盒子 选择 和 多 行 编辑 


在 Visual Studio 2010 中 ， 支 持 对 代码 的 “盒子 选择 (Box Selection) ”， 即 可 以 选择 
屏幕 上 任意 一 个 矩形 区 域内 的 代码 ， 而 不 要 求 是 整 行 选择 。 进 行 盒子 选择 的 操作 方法 是 按 
住 Alt 键 同时 拖 动 鼠标 ， 就 可 以 选中 鼠标 所 划 过 的 矩形 区 域 的 代码 ， 如 图 7.3 所 示 。 


ToewFeature. Student 
Using Systenm. Web; 


Inanespace NevFeature 
{ 
public class Student 


string age: 
string mame 


string school; 
string hobby: 


图 7.3 文本 编辑 器 的 盒子 选择 功能 


使 用 盒子 选择 功能 选中 一 个 矩形 区 域 后 ， 如 果 从 键盘 进行 输入 ， 则 所 输入 的 内 容 会 同 
时 出 现在 所 选择 的 所 有 行 。 利 用 这 个 特性 ， 可 以 完成 一 些 实用 的 功能 。 例 如 ， 在 图 7.3 所 
示 的 Student 类 中 ， 如 果 要 将 每 一 个 字段 的 访问 属性 都 变 成 Public， 则 可 以 用 盒子 选择 方式 
选中 各 个 string 前 面 的 空白 ， 然 后 输入 public， 就 可 以 完成 这 个 任务 。 


7.1.3 ”快速 搜索 


Visual Studio 2010 新 增 了 一 个 快速 搜索 (Quick Search) 功能 ， 可 以 快速 模糊 搜索 文件 
中 的 代码 。 在 代码 编辑 器 中 同时 按 住 Ctl 和 ， 逗号 ) 键 ， 就 可 以 打开 快速 搜索 窗口 ， 在 
其 中 输入 要 查询 的 代码 片断 ， 即 可 以 实时 显示 查找 结果 。 例 如 ， 在 某 段 代码 中 ， 程 序 员 只 
记得 某 段 代码 包含 get 和 date， 却 忘记 了 精确 的 代码 ， 则 可 以 使 用 快速 搜索 窗口 ， 在 其 中 


"和 
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输入 get+ 空 格 +date， 就 可 以 搜索 出 解决 方案 中 同时 包括 get 和 date 的 代码 , 如 图 7.4 所 示 


定位 到 ‘2 x] 
提 过 条件 人 
Te EE 
打扫 到 5 个 P 结 果 四 

ate Datedarseir g, PagedataArement) 个 [ 共 类 型 外]yadLospLL 个 ] 


图 7.4 代码 快速 搜索 
7.1.4 调用 层次 结构 


在 一 个 真实 项 目 中 , 各 个 类 的 方法 之 间 互 相 调用 ,形成 一 个 复杂 的 调用 关系 网 。Visual 
studio 2010 新 增 了 一 个 功能 ， 能 够 显示 方法 之 间 的 调用 层次 结构 ， 帮 助 开发 人 员 理 清 方法 
之 间 的 调用 关系 。 在 代码 视图 中 ， 在 一 个 方法 上 右 击 ， 从 弹出 的 菜单 中 选择 “查看 调用 层 
次 结构 ”选项 ， 如 图 7.5 所 示 。 


图 7.5 调用 层次 结构 菜单 


在 图 7.5 中 选择 了 “查看 调用 层次 结构 ”命令 后 ， 则 打开 如 图 7.6 所 示 的 “调用 层次 
结构 ”窗口 。 窗 口 左 侧 显示 了 调用 所 选择 方法 的 方法 和 被 所 选择 方法 调用 的 方法 。 从 左 侧 
视图 中 选择 一 个 方法 ， 会 在 右 侧 视图 中 显示 调用 此 方法 的 代码 。 


7.1.5 高 亮 显示 引用 


在 Visual Studio 2010 的 代码 编辑 嚣 中， 把 光标 放 在 一 个 标识 符 上 不 管 是 类 、 方 法 、 


.272 。 
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属性 还 是 局 部 变量 ) ， 就 会 高 亮 显示 当前 文件 中 对 于 所 选 标识 符 的 所 有 引用 ， 如 图 7.7 所 
示 ， 高 亮 显 示 了 对 局 部 变量 where 的 所 有 引用 。 


EE 


[EE 到 
EEC 


cDAL [SEE tring county) 


7 区 生 古 称 ， 为 宇 则 所 有 区 且 </p 


pa 
i private ChochhccoamtRoport gotCoutySumary atring Conty) 
“ 曾 孜 | 


bool filter = Istring. a // 是 否 指 定 了 区 县 
rine WE = = 和 和 


7 吉 商 户 号 、 终 卡号 独 相 同 ， 凡生 - 遇 【一 人 次 ) 
string sql = “select count(*) 
+ (select distinct ccrno, term, accne, to_char (tine,’ yyyy-mardd hh2d ) 
se + table + Where 
= Convert. TolInt32 IDbUtility oracle. ExecuteScalar (ComandType. Text, sql)); 
7 
sql = "select sun(nvl (ant,0)), sun(nvl (fee, 0)) , sun(nvl(realant, 0)) fron ”+ table/WharS; 
Var reader = NyDbUtility, oracle, ExecuteReader (CommandType Text, sql); 
证 (reader. Read()) 
{ 


obiect 0 = readerfol 


图 7.7 高 亮 显 示 引 用 


7.2 ASPNET 4.0 新 特性 


Visual Studio 2010 中 包含 的 ASPNET 版 本 为 40，ASP.NET 4.0 新 增 了 控件 静态 ID、 
图 表 控 件 、Web.config 转换 等 新 特性 。 


7.2.1 控件 静态 ID 


在 ASPNET 4.0 以 前 的 版 本 中 ，ASPNET 控件 在 最 终生 成 的 HTML 代码 中 的 ID 
ASPNET 按照 一 定 的 算法 动态 生成 ， 有 时 名 字 会 很 长 ， 例 如 某 页 中 一 个 TextBox 控件 在 客 
户 端 生成 的 ID 为 “ContentPlaceHolderl WebUserControlll MyTextBox1”。 开 发 人 员 无 法 
控制 也， 也 无 法 事先 知道 这 个 ID。 这 种 D 一 方面 使 页 面 看 起 来 代码 较 乱 ， 控 件 名 称 不 直 
观 ， 另 一 方面 也 给 浏览 器 端的 JavaScript 编写 造成 了 不 便 。 

在 ASPNET 4.0 中 , 服务 器 端 控件 有 一 个 ClientIDMode 属性 ,如果 将 其 设置 为 Static， 
则 控件 在 浏览 器 端的 ID 就 是 服务 器 端 设 置 的 ID。 下面 通过 一 个 例子 来 看 控件 静态 ID 与 默 
认 ID 的 区 别 。 

【 例 7-1】 控件 静态 人 D。 

本 例 演示 了 ASP.NET 服务 器 端 控件 默认 的 ID 生成 方式 与 静态 D 的 区 别 ， 以 及 静态 
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ID 对 于 客户 端 JavaScript 编程 的 优势 。 
(1) 创建 一 个 ASPNET 空 项 目 ， 在 项 目 中 添加 一 个 母 版 页 Sitel.master。 
(2) 在 项 目 中 添加 一 个 用 户 控件 WebUserControll.ascx， 在 其 中 添加 两 个 TextBox 控 


件 ; 一 个 使 


默认 ID 生成 方式 ， 另 外 一 个 使 用 静态 DD， 代 码 如 下 : 


<asp:TextBox runat="server" id="TextBox1"> </asp:TextBox> 
<asp:TextBox runat="server" id="TextBox2" ClientIDMode="Static" > 


</asp: 


TextBox> 


(3) 从 母 版 页 Sitel.master 创建 一 个 内 容 页 面 WebForm2.aspx, 在 页 面 上 放置 一 个 刚才 


所 创建 的 上 


户 控件 WebUserControll, 然后 再 放置 两 个 Button 控件 , 一 个 使 用 默认 ID 生成 


方式 ， 男 外 一 个 使 用 静态 ID， 代 码 如 下 : 


<asp:Content ID="Content2" ContentPlaceHolderID="ContentPlaceHolder1l" 
runat="server"> 
<ucl:WebUserControll ID="WebUserControl11"” runat="server" /><br /> 
<asp:Button runat="server"” id="buttonl" Text=" 默 认 ID"” /> 
<asp:Button runat="server" id="button2" ClientIDMode="static" Text=" 


静态 ID"” /> 


</asp: 


Content> 


(4) 在 浏览 器 中 浏览 WebForm2.aspx 页 面 ,并 查看 其 HTML 源 代码 ,可 以 得 到 页 面 最 
终生 成 的 HTML， 代 码 如 下 : 


<div> 


<input name="ctl100$ContentPlaceHolderl$WebUserControlll$TextBoxl" type= 
"text" id="ContentPlaceHolder] WebUserControl11 TextBox1"” /> 

<input name="ctl100$ContentPlaceHolderl$WebUserControlll$TextBox2" type= 
"text" id="TextBox2" /> 

<br /> 

<input type="submit" name="ct100$ContentPlaceHolder1l$buttonl" value=" 默 认 
ID" id="ContentPlaceHolderl buttonl" /> 

<input type="submit" name="ct1l00$ContentPlaceHolder1l$button2" value=" 静 态 
ID" id="button2" /> 

</div> 


从 上 述 


HTML 代码 可 以 看 出 ， 默 认 情 况 下 ，ASPNET 服务 器 端 控件 生成 了 较 长 的 客 


户 端 ID， 而 使 用 静态 ID 的 控件 其 客户 端 卫 与 服务 器 端 ID 相同 。 对 照 HTML 代码 与 


ASPNET 页 面 代码 还 可 以 看 出 ， 默 认 情况 下 ， 控 件 的 客户 端 ID 值 为 其 各 级 容器 控件 的 ID 


与 控件 本 身 


ID 的 组 合 。 例 如 ， 内 容 页 中 Buttonl 控件 的 容器 为 ContentPlaceHolderl 控件 ， 


故 Buttonl 的 客户 端 ID 为 ContentPlaceHolderl button1。 页 面 上 TextBoxl 的 容器 为 用 户 控 
件 WebUserControll， 而 WebUserControll 的 容器 为 ContentPlaceHolder1， 故 TextBoxl 控 


件 的 客户 端 
(5) 如 
实现 难度 和 


ID 为 ContentPlaceHolderl WebUserControlll TextBox1。 
果 要 在 浏览 器 端 使 用 JavaScript 操作 两 个 TextBox 控件 ， 例 如 获取 其 中 的 文本 ， 
工作 量 都 有 较 大 差别 。 对 于 静态 ID 控件 来 说 ,由 于 可 以 明确 知道 其 客户 端 人 D， 


因此 JavaScript 编写 很 简单 ， 只 需 使 用 以 下 代码 即 可 。 


function getValuel() { 
Var V = document .getElementBylId('TextBox2') .value; 


al 
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(6) 要 获得 页 面 中 采用 默认 ID 命令 的 TextBox 控件 的 值 ， 则 需要 采用 较为 复杂 的 方 
法 。 在 JavaScript 中 获得 ASPNET 服务 器 控件 ID 通常 使 用 以 下 代码 。 


var id = '<%=button]l.ClientID%S>"; // 其 中 buttonl 为 ASP.NET 服务 器 控件 


在 本 例 中 ， 由 于 TextBoxl 位 于 用 户 控件 中 ， 是 一 个 private 成 员 ， 所 以 不 能 直接 采用 
上 述 代码 获得 其 ID。 可 以 在 用 户 控件 中 添加 一 个 public 属性 , 通过 此 属性 返回 TextBox 控 
件 的 客户 端 ID， 代 码 如 下 : 

i string textBoxId 


get { return TextBoxl.ClientID; } 


在 WebForm2.aspx 页 面 中 , 通过 访问 此 属性 得 到 TextBox 控件 的 客户 端 ID, 并 应 用 于 
JavaScript 代码 中 。 


function getValue2() { 
Var V = document .getElementById('<%=WebUserControlll .textBoxId%>"'). 
value; 
alert (v); 


7.2.2 图 表 控 件 


ASPNET 4.0 中 的 图 表 控件 (Chart 控件 ) 类似 于 Excel 中 的 图 表 控 件 ， 可 以 根据 数据 
源 生 成 各 种 常见 图 形 报 表 ， 如 条 形 图 、 饼 形 图 等 ， 图 7.8 为 Chart 控件 的 生成 的 两 个 图 表 。 


3- 
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图 7.8 ”Chart 控件 示例 


【 例 7-2】 图 表 控 件 示例 。 

本 例 演 示 了 图 表 控 件 的 简单 使 用 。 

(1) 创建 一 个 ASP.NET Web 应 用 程序 ， 添 加 一 个 页 面 ChartSample.aspx。 

(2) 打开 ChartSample.aspx 页 面 ， 从 工具 箱 的 “数据 ”控件 组 中 找到 “Chart” 控 件 ， 
如 图 7.9 所 示 ， 将 其 拖 动 到 页 面 上 。 

(3) 在 ChartSample.aspx 页 面 中 选中 Chart 控件 ， 单 击 其 右上 角 的 按钮 ， 从 打开 的 智 
能 任务 页 面 中 选择 “新 建 数据 源 ” 选 项 ， 如 图 7.10 所 示 。 

(4) 在 弹出 的 “数据 源 配置 向 导 ” 对 话 框 中 选择 “SQL 数据 库 ”， 如 图 7.11 所 示 。 


a 
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指针 
AccessDataSource 


图 7.9 工具 箱 里 的 Chart 控件 7.10 为 Chart 控件 添加 数据 源 


(5) 在 “数据 源 配置 向 导 ” 的 后 续 步 骤 中 ， 设 置 服务 器 为 本 机 ， 数 据 库 为 Northwind， 
查询 语句 为 “select CategoryName, CategorySales from [Category Sales for 1997]”， 其 中 
Category Sales for 1997 是 Northwind 数据 库 中 一 个 视图 , 其 中 数据 为 1997 年 按照 产品 类 别 


统计 的 销售 额 。 


(6) 在 Chart 控件 的 智能 任务 面板 中 ， 设 置 其 横 坐 标 和 纵 坐标 使 用 的 数据 来 源 ， 其 中 
横 坐 标 为 产品 类 别名 称 CategoryName， 纵 坐标 为 销售 金额 ， 如 图 7.12 所 示 。 


二 3 


7.11 配置 数据 源 


(7) 在 浏览 器 中 查看 ChartSample.aspx 页 面 ， 运 行 界面 如 图 7.13 所 示 。 


7.12 设置 坐标 数据 


| 


De Aeasest Owewres 


nple tsyx 全 


图 7.13 ”ChartSample 运行 界面 
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7.2.3 Web 配置 文件 转换 


ASPNET 应 用 程序 的 配置 文件 Web.config 中 通常 包含 一 些 环境 参数 ， 如 连接 字符 串 
等 。 当 ASP.NET 应 用 程序 运行 环境 发 生 改变 时 ，Web.config 里 面 的 内 容 通常 也 需要 进行 调 
整 。 例 如 ， 在 开发 环境 与 部 署 环境 下 ， 数 据 库 连 接 字符 串 不 同 ， 自 定义 错误 也 不 相同 。 在 
ASP.NET 4.0 以 前 ， 需 要 通过 手工 修改 Web.config 文件 内 容 的 方式 实现 这 个 功能 。 在 
ASPNET 4.0 中 ， 引 入 了 一 个 新 功能 Web 配置 文件 转换 (Web.config transformation〉 可 以 
自动 完成 这 个 功能 。 

Web 配置 文件 转换 的 原理 为 在 ASPNET 项 目 中 定义 一 个 基本 的 Web 配置 文件 ， 再 定 
义 若干 个 转换 文件 ， 每 个 转换 文件 对 应 一 种 特定 的 配置 环境 ， 最 常见 的 有 两 种 环境 ， 分 别 
是 调试 环境 (Debug) 和 部 署 环 境 (Release) 。 在 转换 文件 中 定义 一 些 转换 规则 ， 对 基本 
Web 配置 文件 中 的 内 容 进 行 转换 (如 删除 、 替 换 等 ) ， 从 而 得 到 一 个 新 的 Web 配置 文件 。 
生成 ASP.NET 项 目 时 ， 选 择 一 个 合适 的 配置 环境 (如 Debug、Release 等 ) ， 就 可 以 得 
到 适应 于 此 环境 的 Web.config 文件 。 

【 例 7-3】 Web 配置 文件 转换 。 

本 例 演示 如 何 利用 Web 配置 文件 转换 功能 生成 不 同 环境 下 的 Web 配置 文件 。 

(1) 创建 一 个 ASP.NET Web 应 用 程序 。 

(2) 从 菜单 中 选择 “生成 ”|“ 配 置 管理 器 ”命令 ， 则 打开 “配置 管理 器 ”窗口 ， 如 图 
7.14 所 示 。 

(3) 在 图 7.14 所 示 的 配置 管理 器 中 ， 可 以 看 到 目前 已 经 存在 两 个 配置 Debug 和 
Release， 这 两 个 配置 是 ASP.NET 自动 生成 的 。 从 顶部 的 下 拉 列 表 框 中 选择 “新 建 ”选项 ， 
则 弹出 如 图 7.15 所 示 的 “新 建 解决 方案 配置 ”对 话 框 ， 在 其 中 输入 名 称 MyConfig， 并 单 
击 “ 确 定 ” 按 钮 。 


I 


活动 解决 方案 配置 C); 活动 解决 方案 平台 全) 
Dobue a [sy ce 了 


新 建 解 决 方案 配置 Hx 


图 7.14 配置 管理 器 图 7.15 新 建 解决 方案 配置 


(4) 在 解决 方案 资源 管理 器 中 ， 右 击 Web.config 文件 ， 从 弹出 的 菜单 中 选择 “添加 配 
置 转换 ”命令 , 则 会 在 解决 方案 中 添加 一 个 新 的 文件 Web.MyConfig.config, 其 中 MyConfig 
为 刚才 所 添加 的 解决 方案 配置 名 称 ， 如 图 7.16 所 示 。 

(5) 为 了 验证 配置 文件 转换 功能 ， 在 项 目 中 分 别 为 默认 的 配置 文件 web.config 和 各 个 
配置 转换 文件 添加 不 同 的 内 容 。 首 先 打开 默认 配置 文件 Web.config， 在 其 中 输入 两 个 连接 
字符 串 ， 代 码 如 下 : 


a 
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<connectionStrings> 
<add name="dbl" connectionString="Data Source=.; Initial Catalog= 
DefaultDb; Integrated Security=True"/> 
<add name="db2" connectionString="Data Source=.; Initial Catalog= 
DefaultDb; Integrated Security=True"/> 

</connectionStrings> 


(6) 打开 Web.Debug.config 文件 ， 这 是 一 个 配置 转换 文件 ， 其 中 包含 了 一 些 配置 文件 
替换 规则 。 在 Web.Debug.config 文件 中 输入 以 下 内 容 。 
<connectionStrings> 
<add name="db1" 
connectionString="Data Source=DebugServer; Initial Catalog=Northwind; 
Integrated Security=True" 
xdt:Transform="SetAttributes" xdt:Locator="Match (name)"/> 
</connectionstrings> 
上 述 代 码 定义 了 对 ConnectionString 的 一 个 转换 规则 ，xdt:Transform="SetAttributes" 表 
示 转 换 方式 为 设置 属性 ，xdt:Locator="Match(name)" 表 示 匹 配 规则 为 名 称 相同 。 整 个 代码 的 
含义 为 查找 name 值 相同 的 连接 字符 串 ， 并 用 此 处 提供 的 新 值 设 置 原 有 属性 。 


各 提示 : ASPNET 支持 的 配置 文件 转换 方式 有 多 种 ， 除 了 SetAttributes 外 ， 还 有 Replace、 
Remove、Insert 等 ， 分 别 表示 替换 、 删 除 和 插入 ， 详 情 可 参见 MSDN。 


(7) 打开 Web.MyConfig.config 配置 转换 文件 ， 在 其 中 输入 以 下 内 容 : 


<connectionStrings> 
<add name="db1" 
connectionString="Data Source=MyServer; Initial Catalog=Northwind; 
Integrated Security=True" 
xdt:Transform="SetAttributes" xdt:Locator="Match (name)"/> 
</connectionstrings> 


(8) 从 Visual Studio 工具 栏 中 选中 Debug 作为 当前 配置 ， 然 后 从 菜单 中 选择 “生成 ” 
1 “发布 ”命令 ， 则 弹出 如 图 7.17 所 示 的 “发 布 Web” 对 话 框 。 在 对 话 框 的 “发 布 方法 ” 
下 拉 列 表 框 中 选择 “文件 系统 ”， 并 在 “目标 位 置 ”下 拉 列 表 框 中 设置 一 个 合适 的 发 布 路 
径 ， 然 后 单 击 “ 发 布 ” 按 钮 以 发 布 项 目 。 


发 布 Yeb 
发 布 配置 文件 (H) | 
lL 岂 方案 资源 管理 器 配置 文件 1 重 命名 四 ) 上 出 
避 |3 回 | 第 二 “机上 昭和 ”的 “打包 /发 币 Nob" 和 “打包 /发 
辐 解决 方案 “ysS2010”(1 个 项 目 ) 
ey en 
国 Properties SERIES 
加 引用 发 布 
9 EE] chartSanple aspx 生成 配置 :Debug 
Wogan 校 用 生成 配 寻 生理 状 示 更 改 配 于 
9 sitet Naster 
Student, cs 发 布 方法 如，| 文 件 系统 = 
上 E 部 署 


Ee YWeb. edue config 
Yeb. lyConfig, config 
a Web. Release. config 
可 WebFornl. aspx 
昌国 WebF orn2. aspx 
WebUserControll. ascx 


图 7.16 新 添加 的 配置 转换 图 7.17 “发 布 Web” 对 话 框 
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(9) 发 布 成 功 后 ， 找 到 发 布 的 目标 位 置 ， 打 开 Web config 文件 ， 可 以 看 到 连接 字符 中 
内 容 如 下 : 


<connectionStrings> 
<add name="dbl" connectionString="Data Source=DebugServer; Initial 
Catalog=Northwind; Integrated Security=True"/> 
<add name="db2" connectionString="Data Source=.; Initial Catalog= 
DefaultDb; Integrated Security=True"/> 

</connectionstrings> 


对 照 以 上 连接 字符 串 内 容 和 开发 环境 中 的 各 个 连接 字符 串 可 以 看 出 ， 开 发 环境 中 
Web.config 文件 中 第 1 个 连接 字符 串 被 蔡 换 为 Web.Debug.config 的 内 容 ， 而 第 2 个 连接 字 
符 串 未 发 生 改 变 。 

(10) 在 Visual Studio 开发 环境 中 选中 MyConfig 作为 当前 配置 ， 再 次 发 布 网 站 ， 并 查 
看 发 布 后 的 Web.config 文件 ， 得 到 以 下 内 容 。 


<connectionStrings> 
<add name="dbl" connectionString="Data Source=MyServer; Initial Catalog= 
Northwind; Integrated Security=True"/> 
<add name="db2" connectionString="Data Source=.;Initial Catalog= 
DefaultDb; Integrated Security=True"/> 

</connectionstrings> 


7.3”C#4.0 新 特性 


在 Visual Studio 2010 中 ，C# 语 言 升级 到 最 新 版 本 4.0。C# 4.0 中 增加 了 动态 类 型 、 命 
名 和 可 选 参数 、 协 变 逆 变 等 新 功能 。 


7.3.1 动态 类 型 


在 C# 4.0 以 前 的 版 本 中 ， 变 量 都 是 静态 类 型 ， 即 变量 的 类 型 在 编译 阶段 就 能 够 确定 。 
即使 使 用 var 关键 字 声 明 的 变量 ， 在 声明 的 同时 必须 提供 一 个 默认 值 ， 根 据 这 个 值 就 可 以 
确定 出 变量 的 类 型 ， 因 此 这 个 变量 实际 上 也 是 静态 变量 。 

所 谓 动态 类 型 ， 是 指 无 法 在 编译 阶段 确定 变量 类 型 ， 只 有 等 到 运行 时 才能 获得 变量 实 
际 类 型 。 动 态 类 型 的 好 处 是 能 够 编写 更 加 灵活 、 强 大 、 简 洁 的 代码 。 其 劣势 在 于 : (1) 由 
于 编译 阶段 无 法 知道 变量 的 类 型 ， 因 此 不 进行 类 型 安全 检查 ;， 〈2) 性 能 要 比 静 态 类 型 低 ; 
(3) 无 法 在 Visual Studio 开发 环境 中 显示 智能 提示 。 由 于 以 上 原因 ， 除 非 必要 ， 和 否则 不 要 
使 用 动态 类 型 。 

C# 4.0 中 使 用 dynamic 关键 字 定 义 动态 类 型 。 动 态 类 型 的 意味 着 写 代码 时 不 能 确定 这 
个 类 型 是 什么 ， 也 不 知道 这 个 类 型 有 什么 方法 和 属性 。 对 动态 类 型 可 以 执行 任意 操作 : 赋 
值 或 调用 任何 方法 、 访 问 任何 属性 、 作 为 参数 传递 给 函数 等 。 如 下 面 的 代码 所 示 。 


dynamic myDynamic = getDynamic(); // 此 方法 返回 一 个 不 能 确定 类 型 的 对 象 
myDynamic.anyMethod ("Hello"); // 调 用 方法 

myDynamic.propertyl = myDynamic.property2; // 访 问 属性 

int Quux = myDynamic[0]; // 索 引 器 
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int Qux = myDynamic (12,5) // 作 为 方法 (委托 ) 使 用 
someMethod (myDynamic); // 作 为 方法 参数 传递 


于 myDynamic 是 动态 类 型 ,在 编译 时 无 法 知道 其 类 型 、 属 性 、 方 法 ,无 法 进行 类 型 
安全 检查 ， 所 以 以 上 代码 会 编译 通过 。 如果 myDynamic 不 支持 某 个 属性 或 者 方法 , 或 者 其 
类 型 与 函数 所 期 待 的 参数 类 型 不 一 致 ， 只 有 等 到 程序 运行 时 才 会 发 现 这 个 错误 。 


7.3.2 命名 和 可 选 参数 


在 开发 项 目 过 程 中 ， 有 时 需要 编写 一 个 接受 多 个 方法 的 参数 ， 例 如 ， 某 销售 管理 系统 
需要 根据 一 系列 条 件 查询 销售 情况 ， 过 滤 条 件 为 销售 日 期 、 商 品类 别 、 商 品 编号 、 销 售 城 
市 的 任意 组 合 ， 如 果 指 定 了 某 个 条 件 ， 则 根据 这 个 条 件 进 行 过 滤 ， 和 否则 不 对 此 条 件 进 行 限 
制 。 如 果 用 一 个 方法 实现 此 功能 ， 则 此 方法 需要 接受 上 述 所 有 查询 条 件 作为 参数 ， 代 码 
如 下 : 


/// <summary> 

/// 根据 用 户 指定 条 件 查询 销售 情况 

/// </summary> 

/// <param name="from"> 查 询 开始 时 间 </param> 

/// <param name="to"> 查 询 截止 时 间 </param> 

/// <param name="categoryId"> 商 品类 别 ID</param> 
/// <param name="productId"> 商 品 ID</param> 

/// <param name="cityId"> 销 售 城市 ID</param> 

/// <returns> 查 询 得 到 的 销售 记录 列表 </returns> 


static List<SaleRecord> select( DateTime from, DateTime to, string 
categoryId, 

string productId, string cityId) 
ee 

在 调用 上 述 方法 执行 查询 时 ， 很 多 情况 下 并 不 需要 根据 所 有 条 件 进行 查询 ， 而 只 需要 
根据 部 分 条 件 进行 查询 。 在 C#4.0 以 前 ， 如 果 要 调用 这 个 方法 ， 必 须 为 所 有 参数 都 指定 一 
个 值 ， 即 使 不 需要 根据 这 个 条 件 进行 查询 。 例 如 ， 如 果 要 根据 销售 日 期 和 销售 城市 进行 查 
询 ， 则 需要 使 用 以 下 代码 。 

var list = select (DateTime.Parse("2010-1-1"), DateTime.Parse ("2010-1-10"), 

Ra nuUlle MUOOLSN 

在 C#4.0 中 ， 人 允许 为 方法 的 参数 指定 默认 值 ， 则 此 参数 成 为 可 选 参数 ， 调 用 方法 时 可 
以 不 必 给 这 个 参数 传 值 。C# 4.0 还 支持 命名 参数 ， 调 用 方法 时 可 以 通过 参数 名 称 为 某 个 特 
定 参数 传 值 。 
全 提示 : 可 选 参数 必须 位 于 方法 参数 列表 的 最 后 ， 即 在 可 选 参 数 后 面 不 允许 出 现 不 可 先 

参数 。 
利用 C#4.0 的 可 选 参数 ， 上 述 代码 可 以 修改 成 以 下 形式 。 
static List<SaleRecord> select( DateTime from, DateTime to, string 


categoryId=null, 
string productId=null, string cityId=null) 
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如 果 要 查询 某 个 时 间 段 的 销售 情况 ， 则 可 以 使 用 以 下 代码 。 


var list = select (DateTime.Parse("2010-1-1"), DateTime.Parse("2010-1 
=1020) 


如 果 要 查询 某 个 时 间 段 某 城市 的 销售 情况 ， 则 可 以 使 用 以 下 代码 。 


var list = select (DateTime.Parse("2010-1-1"), DateTime.Parse ("2010-1-10"), 
Cet a rs np 


上 述 代 码 中 的 cityId:"1010" 表 示 为 方法 的 cityId 参数 传递 的 值 为 1010， 这 就 是 C# 4.0 
使 用 命名 参数 的 语法 。 


7.3.3 协 变 和 逆 变 


在 C# 4.0 中 引入 了 协 变 〈Covariance) 和 逆 变 (Contravariance〉 的 概念 以 增强 泛 型 接 
口 和 委托 。 在 C# 中 ， 将 一 个 IList<string> 类 型 转换 为 IList<objec 人 是 不 允许 的 ， 如 下 代码 
在 编译 时 会 出 错 。 

IList<string> words = new List<string> { "this", "is", "a", "string", 


"arcray™ }s 
IList<object> objects = words; 


上 述 代码 编译 时 会 提示 “无 法 将 类 型 IList<string> 隐 式 转换 类 IList<object>”， 如 果 确 
实 要 实现 转换 ， 则 需要 一 个 强制 类 型 转换 。 有 些 刚 接触 C# 的 程序 员 可 能 认为 String 可 以 隐 
式 转换 为 Object， 所 以 String 的 列表 也 就 可 以 隐 式 转换 为 Object 的 列表 ， 这 个 想法 是 错误 
的 。C# 之 所 以 不 允许 将 IList<string> 转 换 为 IList<objec 人 是 出 于 类 型 安全 的 原因 。 如 果 人 允许 
转换 ， 则 考虑 以 下 代码 。 


IList<string> words =newList<string>("this"，"is"，"a"n，"String",narray") 7; 


IList<object> objects = numbers; // 假 如 可 以 转换 成 功 
objects[0] = 50; 

objects[1] = 123.546; 

objects[2] = new SaleRecord(); 

string s = words[2]; // 出 错 


假如 可 以 将 IList<string> 转 换 为 IList<object>， 由 于 IList 是 引用 类 型 ， 上 述 代码 中 的 
objects 和 words 其 实 是 同一 个 对 象 。 通 过 objects 可 以 向 列表 中 添加 任意 类 型 的 对 象 , 然后 
通过 words 试图 得 到 一 个 string 类 型 时 就 会 出 错 。 

虽然 IList<string> 不 可 以 转换 为 IList<object>, 但 是 IEnumerable<string> 却 可 以 转换 为 
IEnumerable<object>， 代 码 如 下 : 


IEnumerable<string> words = new List<string> { "this", "is", "a", "string", 

四 全 

和 objects = words; 

C# 人 允许 从 IEnumerable<string> 到 [Enumerable<objec 人 的 转换 ， 是 由 于 IEnumerable<T> 
接口 没有 修改 数据 的 方法 ， 不 会 出 现 前 述 的 类 型 安全 问题 。 这 种 从 派生 类 泛 型 接口 向 基 类 
泛 型 接口 的 转换 称 为 协 变 。C# 是 如 何 知 道 什么 情况 下 允许 协 变 (如 IEnumerable<T> 接 口 ) 
哪些 情况 下 不 允许 (如 IList<T>) 呢 ? 这 是 通过 泛 型 定义 实现 的 。 查 看 这 两 个 泛 型 接口 的 


=281 


第 2 篇 开发 工具 与 第 三 方 框架 


定义 ， 可 以 得 到 以 下 代码 。 

public interface IEnumerable<out T> : IEnumerable 

es 
public interface IList<T> : ICollection<T>, IEnumerable<T>, IEnumerable 
eh 
上 述 代码 可 以 看 出 ， 在 下 numerable 泛 型 接口 的 类 型 参数 工 前 面 有 一 个 out 关键 字 ， 
而 IList 泛 型 接口 则 没有 这 个 out 关键 字 。 泛 型 接口 类 型 参数 工 前 面 的 out 表示 工具 出 现在 
接口 的 输入 位 置 〈 如 方法 返回 值 ) ， 而 不 会 出 现在 输入 位 置 《如 方法 输入 参数 ) 。 像 这 种 
在 类 型 参数 前 有 out 关键 字 的 泛 型 接口 允许 协 变 。 

与 协 变相 对 的 概念 称 为 逆 变 ， 即 将 一 个 基 类 型 的 泛 型 接口 转换 为 派生 类 型 的 泛 型 接 
口 ， 代 码 如 下 : 


IComparer<object> objectComparer=getObjectComparer (); 

IComparer<string> stringComparer=getStringComparer (); 

stringComparer = objectComparer; // 道 变 

上 述 代码 的 第 三 行 ,将 一 个 IComparer<objec 人 > 类 型 隐 式 转换 为 IComparer<string> 类 型 。 
这 个 转换 与 通常 所 理解 的 类 型 转换 有 点 矛盾 ， 通 常 允 许 从 派生 类 向 基 类 的 隐 式 转换 ， 基 类 
却 不 能 隐 式 转换 为 派生 类 型 。 对 于 IComparer<T> 接 口 来 说 ， 由 基 类 接口 向 派生 类 接口 转换 
是 有 意义 的 ， 如 代码 中 一 个 可 以 比较 Object 类 型 的 比较 器 ， 自 然 可 以 成 为 比较 string 类 型 
的 比较 器 。 查 看 IComparer 泛 型 接口 的 定义 ， 可 得 到 如 下 代码 。 

public interface IComparer<in T> 

el 

通过 上 述 代 码 可 看 出 ,在 IComparer 泛 型 接口 的 类 型 参数 工 前 面 有 一 个 ip 关键 字 , 表 
示 在 此 泛 型 接口 中 类 型 参数 工具 允许 出 现在 输入 位 置 〈 如 方法 的 输入 参数 ) ， 而 不 允许 出 
现在 输出 位 置 〈 如 方法 返回 值 ) 。 像 这 种 在 类 型 参数 前 有 in 关键 字 的 泛 型 接口 允许 逆 变 。 


外 提示 : 用 于 指示 协 变 和 逆 变 的 in、out 关键 字 只 能 用 于 修饰 泛 型 接口 和 委托 的 类 型 参数 ， 
不 能 用 于 类 、 结 构 、 方 法 等 。 所 修饰 的 类 型 必须 是 引用 类 型 ， 不 能 是 值 类 型 。 


74 小 结 


伴随 着 Visual Studio 2010 的 发 布 ，NET Framework、ASP.NET、C# 都 升级 到 了 4.0 版 
本 ,增加 了 一 些 新 功能 。 本 章 介 绍 了 Visual Studio 2010 IDE 新 增 功能 、ASP.NET 的 新 特性 
和 C#4.0 的 新 语法 。 
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数据 访问 一 直 是 大 多 数 应 用 程序 的 一 个 非常 重要 的 工作 。 为 了 使 数据 访问 的 编码 变 得 
简单 、 高 效 ， 软件 开 发 领域 不 断 推 出 新 的 数据 访问 框架 和 产品 。 用 于 .NET 领域 的 数据 访问 
框架 有 Enterprise Library、ActiveRecord、NHibermate、SubSonic 等 ，ASP.NET 中 也 包含 许 
多 数据 访问 相关 的 控件 ， 如 各 种 数据 源 控件 和 数据 绑 定 控件 。 微 软 公司 推出 的 最 新 数据 访 
问 技术 为 语言 集成 查询 (Language-Integrated Query 简称 LINQ ) 和 实体 框架 (Entity 
Framework 简称 EF) 。 


8.1 C#x LINQ 的 支持 


LINQ 提供 了 一 种 跨 各 种 数据 源 和 数据 格式 使 用 数据 的 一 致 模型 。 在 LINQ 查询 中 ， 
始终 使 用 对 象 而 非 针对 某 种 具体 数据 源 的 操作 命令 ， 可 以 使 用 相同 的 基本 编码 模式 来 查询 
和 转换 XML 文档 、SQL 数据 库 、ADO.NET 数据 集 、.NET 集合 中 的 数据 及 对 其 有 LINQ 
提供 程序 可 用 的 任何 其 他 格式 的 数据 。LINQ 是 一 种 全 新 的 数据 查询 方式 ，C# 语 言 为 全 面 
支持 LINQ 增加 了 一 些 新 的 语法 和 功能 ， 本 节 将 对 此 进行 介绍 。 


8.1.1 对象 初始 化 器 


在 编程 过 程 中 经 常用 到 的 一 个 功能 就 是 在 创建 对 象 时 为 对 象 的 某 些 属性 赋值 。C# 的 对 
象 初始 化 器 (Object Initializer) 可 以 很 好 地 实现 这 一 功能 。 其 语法 如 下 : 
new 类 名 () {属性 1= 值 1， 属 性 2= 值 2，...， 属 性 N= 值 N}; 


下 面 通过 例子 来 具体 讲解 如 何在 创建 类 的 实例 时 为 其 属性 赋值 。 有 一 个 Student 类 ， 
定义 如 下 : 
class Student 
{ 
public string name { get; set; } 
public int age { get; set; } 
} 
在 创建 Student 类 的 实例 时 ， 可 以 给 name 属性 、age 属性 或 者 二 者 同时 赋值 ， 代 码 
如 下 : 


第 2 篇 开发 工具 与 第 三 方 框架 


// 创 建 一 个 Student 类 的 实例 ， 并 设置 其 name 属性 

Student sl = new Student(){ name="UJohn"}7 

上 面 一 行 代码 的 功能 相当 于 以 下 代码 : 

Student sl = new Student(); 

sl.name = "Jonh"; 

也 可 以 在 创建 类 的 实例 时 同时 为 多 个 属性 赋值 ， 代 码 如 下 : 


Student s2 = new Student {age=20,name="Mary™" }; 


8.1.2 隐 式 类 型 


使 用 LINQ 进行 查询 时 ， 很 多 时 候 编程 人 员 不 容易 判断 某 个 查询 返回 的 类 型 ， 或 者 类 
型 名 称 太 长 不 方便 写 出 来 ，C# 中 引入 了 隐 式 类 型 这 个 概念 以 解决 此 问题 。 在 C# 中 可 以 使 
用 var 关键 字 来 声明 隐 式 类 型 的 局 部 变量 ， 其 语法 如 下 : 

var 变量 名 = 初始 值 ; 


通过 使 用 var 关键 字 ， 就 可 以 很 方便 地 表示 LINQ 的 查询 结果 。 如 下 代码 所 示 ， 等 号 
右面 的 所 有 代码 为 一 个 LINQ 查询 语句 ， 这 个 查询 语句 返回 一 个 不 能 知道 名 称 的 类 型 〈 称 
为 匿名 类 型 )， 这 种 情况 下 必须 使 用 var 关键 字 才 能 保存 这 个 返回 值 。 

var query= from p in products 


where p.price > 20 && p.price < 100 
select p; 


用 var 关键 字 声 明 的 变量 可 以 被 初始 化 成 任何 类 型 的 值 ， 如 下 面 的 代码 所 示 。 
// 用 int 值 初始 化 var 变量 


var varl = 21; 


// 用 string 值 初始 化 var 变量 


Var Var2 = "some text"; 


// 用 DateTime 值 初始 化 var 变量 


var var3 = DateTime.Now; 


// 用 数组 初始 化 var 变量 


var var4=new object[10]; 


// 把 var 变量 初始 化 为 列表 
var var4 = new List<string>(); 
在 使 用 var 关键 字 定 义 变量 时 ， 需 要 注意 以 下 两 点 。 
(1) 使 用 var 关键 字 定义 的 变量 必须 被 初始 化 。 如 下 面 的 代码 是 错误 的 。 
static void Main(string[] args) 
{ 
var num; //var 变量 未 初始 化 
num = 10; 
} 
(2) 使 用 var 关键 字 只 能 声明 局 部 变量 (包括 在 for、foreach、using 语句 中 使 用 的 变 
量 )， 而 不 能 用 于 声明 其 他 变量 ， 如 下 面 的 例子 所 示 。 
class VarSample 
{ 
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// 错 误 ，var 不 能 声明 类 中 的 字段 
var name = "名 字 "™; 
// 错 误 ，var 不 能 声明 方法 参数 


private void methodl (var i) 


Console.WriteLine (i); 
} 
private void method2 () 
// 正 确 ，var 可 用 于 for 循环 中 声明 计数 变量 
EC (var 1 OF i < OA 
Console.WriteLine (i); 
int[] nums = new intll IT 20 37 4 J 
// 正 确 ，var 可 用 于 foreach 语句 中 的 循环 变量 
foreach (var num in nums) 
. 
Console.WriteLine (num); 
} 
// 正 确 ，var 可 用 于 using 语句 中 的 变量 声明 


using (var file =new System.I0.StreamWriter("e:\\temp.text")) 


{ 
file.WriteLine ("some text"); 


} 
l 


(3) var 变量 和 object 类 型 变量 完全 不 同 。 使 用 object 类 型 声明 的 变量 是 弱 类 型 ， 可 以 
被 赋予 任何 类 型 的 值 ， 而 使 用 var 关键 字 声 明 的 变量 与 普通 变量 一 样 ,仍然 是 强 类 型 变量 。 
var 变量 被 初始 化 时 ， 其 类 型 即 被 确定 。 

(4) var 变量 与 dynamic 变量 完全 不 同 。var 变量 是 一 种 静态 类 型 的 变量 ， 而 dynamic 
是 一 种 动态 类 型 变量 。 静态 类 型 可 以 执行 类 型 检查 、 给 出 智能 提示 ,而 动态 类 型 则 不 可 以 。 
在 Visual Studio 开发 环境 中 ， 智 能 识别 系统 能 够 识别 var 变量 的 类 型 ， 从 而 给 出 成 员 提示 ， 


如 图 8.1 所 示 。 


袁 Linqsaple Proerm ~| YNain (string[] args) | 
{ 
war varl = DateTine. Now; 
varl. 


! § Add 
! 9 
class Product 
Y Addlours 


public st 9 Aadlilliseconds 

public st AddMinutes 

public in OAdMonths 

public do 兆 Addseconds 

public st = hadricks ) 

了 ie st Marewrs 村 tsGnt count) 


Random r = new Random(DateTine. Now. Millisecond) ; 


图 8.1 var 变量 为 强 类 型 变量 


从 图 8.1 中 可 以 看 出 , varl 是 一 个 用 var 声明 的 变量 , 并 且 被 初始 化 成 DateTime 类 型 ， 
在 此 后 使 用 varl 变量 时 ，Visual Studio 能 够 自动 识别 出 varl 为 类 型 并 列 出 DateTime 的 


成 员 。 
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于 var 变量 是 强 类 型 变量 ， 所 以 在 var 变量 被 初始 化 后 ， 不 能 再 给 此 变量 赋 其 他 类 
型 不 兼容 的 值 ， 如 下 面 的 代码 所 示 。 
//varl 被 初始 化 为 DateTime 类 型 


Var Varl = DateTime .Now7 

// 下 面 语句 是 错误 的 ， 试 图 用 string 值 赋 给 DateTime 变量 
varl = "尝试 赋 String 类 型 "; 

//var2 被 初始 化 为 int 类 型 

var var2 = 10; 

// 以 下 语句 合法 ， 并 且 var3 也 是 int 类 型 

Var var3 = var2; 

// 错 误 ， 试 图 把 DateTime 值 赋 给 int 类 型 变量 


var3 = varl; 


外 提示 : 用 var 关键 字 声 明 的 隐 式 类 型 变量 是 强 类 型 变量 ， 在 声明 时 必须 赋值 ， 而 且 其 类 
型 一 旦 确定 不 能 更 改 。 


8.1.3 匿名 类 型 


C# 的 匿名 类 型 是 指 没 有 名 字 的 类 型 。 匿 名 类 型 是 一 个 类 ， 直 接 继承 于 object 类 。 匿 名 
类 型 没有 单独 明确 的 类 的 定义 ， 而 是 在 创建 匿名 类 的 实例 时 隐 式 创建 的 类 型 。 匿 名 类 型 的 
成 员 是 一 组 可 以 读 写 的 属性 。 

下 面 的 代码 创建 了 一 个 匿名 类 型 和 此 类 型 的 一 个 实例 。 


var sl = new { id="112",name="John", age=30}; 


上 述 代码 创建 了 一 个 具有 3 个 读 写 属性 的 匿名 类 ， 并 生成 此 类 的 一 个 实例 。 这 行 代码 
在 功能 上 相当 于 以 下 代码 。 
// 定 义 一 个 类 


class AnonymousClassl 

1 
public string id { get; set; } 
public string name { get; set; } 
public int age { get; set; } 


小 
// 创 建 此 类 的 实例 


AnonymousClass] sl = new AnonymousClassl() { id = "112", name = "John", age 

= 0 

在 同一 个 程序 中 , 如 果 两 段 创建 匿名 类 型 对 象 的 代码 具有 相同 的 参数 (包括 参数 类 型 、 
名 称 和 顺序 ), 那么 这 两 段 代码 所 创建 的 匿名 对 象 就 属于 同一 个 匿名 类 型 , 否则 二 者 就 是 不 
同 的 匿名 类 型 。 例 8-1 演示 匿名 类 型 的 使 用 。 

【 例 8-1】 使 用 匿名 类 型 。 


static void Main(string[] args) 


Type typel,type2,type3; //3 个 Type 变量 
var sl = new { id="101",name="John",age=20}; // 创 建 一 个 匿名 类 型 对 象 
typel = s1.GetType()2 // 得 到 并 输出 对 象 的 类 型 名 称 


Console -WriteLine ("typel1:\t"+typel1.Name) 7 
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// 可 以 像 访问 普通 类 型 一 样 访 问 匿名 类 型 的 属性 
string id=sl1.id; 
string name=sl1 .name; 
int age=sl .age; 
Console.WriteLine(string.Format ("id:{0};name: {1};age: {2}",id,name,ag 
e)); 
var 52 = new { id = "102", name = "Mike", age = 25}; 
// 创 建 一 个 匿名 类 型 对 象 

//s1l 和 s2 具有 相同 的 成 员 ， 因 此 属于 相同 的 匿名 类 型 
type2 = s2.GetType(); 
Console.WriteLine ("type2:\t"+type2.Name); 
if (typel == type2) 

Console.WriteLine ("typel 和 type2 相同 ") ; 


s2 = s1; // 相 同 匿 名 类 型 的 对 象 之 间 可 以 赋值 
} 
var s3 = new { id = "101",age = 20,name = "John"}; 

// 创 建 另外 一 个 匿名 类 型 对 象 

// 由 于 参数 不 同 〈 顺 序 不 同 ) ，s3 和 s1 不 属于 同一 匿名 类 型 
type3 = s3.GetType(); 
Console.WriteLine ("type3:\t"+type3.Name); 
if (typel != type3) 
{ 

Console.WriteLine ("sl 和 s3 不 同 "); 
} 


Console.ReadLine (); 


. 

上 述 代 码 的 输出 结果 如 下 : 

typel: <>f AnonymousType0 `3 
id:101;name:John;age:20 
type2: <>f_ AnonymousType0`3 
typel 和 type2 相同 

type3: <>f _ RnonymousTYPel 3 
sl 和 s3 不 同 


8.1.4 扩展 方法 


类 上 


C# 中 的 扩展 方法 允许 向 现 有 类 中 添加 方法 而 不 用 修改 现 有 类 的 代码 。 甚 至 当 没 有 某 个 
骸 源 代码 ， 也 可 以 通过 扩展 方法 向 类 中 添加 方法 。 在 C# 语 言 规范 中 ,微软 公司 建议 应 该 


慎重 使 用 扩展 方法 ， 只 有 在 用 普通 方法 无 法 完成 功能 时 ， 才 应 考虑 使 用 扩展 方法 。 


扩展 函数 只 能 声明 在 静态 类 (static class) 中 ,扩展 函数 与 普通 静态 函数 的 区 别 在 于 其 


方法 的 第 一 个 参数 前 面 有 一 个 this 关键 字 ， 这 个 this 关键 字 是 扩展 函数 的 标志 。this 关键 
字 后 面 的 类 型 为 要 扩展 的 类 型 。 声 明 扩展 方法 的 语法 如 下 : 


static class 封装 类 名 
[访问 修改 符 ] 返回 类 型 扩展 方法 名 (this 扩展 类 名 变量 名 0 
[, 类 型 1 变量 1， 类 型 2 变量 2，… ,类 型 n 变量 n]) 
{ 
方法 体 
a 
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其 中 封装 类 名 是 扩展 方法 的 容器 ， 可 以 是 任意 合法 标识 符 。 扩 展 类 名 是 要 被 扩展 的 类 
名 ， 即 要 在 其 中 添加 方法 的 类 。this 变量 后 面 的 变量 为 方法 参数 列表 。 在 定义 了 扩展 方法 
以 后 ， 扩 展 类 名 所 表示 的 类 型 就 增加 所 定义 的 扩展 方法 。 下 面 通过 一 个 例子 来 看 扩展 方法 
的 使 用 。 

【 例 8-2】 扩展 方法 。 

假设 要 为 NET Framework 中 的 string 类 型 添加 一 个 isInt 方 法, 判断 某 个 字符 串 是 否 是 
个 整数 ， 以 及 是 否 为 正 整 数 。 

(1) 创建 一 个 控制 台 应 用 程序 。 

(2) 在 项 目 中 添加 一 个 类 ， 重 命名 为 StringExtension， 在 class 关键 字 前 面 加 上 static 
关键 字 ， 把 类 声明 成 静态 类 。 代 码 如 下 : 


static class StringExtension 


(3) 在 类 中 添加 一 个 string 类 的 扩展 isInt， 用 来 判断 一 个 字符 串 是 否 整 数 。 代 码 如 下 : 


/// <summary> 

/// 判断 字符 串 是 否 一 个 整数 或 正 整 数 

/// </summary> 

/// <param name="text"> 要 判断 的 字符 串 </param> 

/// <param name="positive"> 是 否 要 求 正 数 </param> 

public static bool isInt(this string text, bool positive=false) 
! 


int n; 
if (int.TryParse (text, out n)) // 如 果 能 把 字符 串 解 析 为 int 类 型 
{ 

if (positive) // 如 果 要 求 正 数 


{ 
return n > 0; // 判 断 解析 得 到 的 整数 是 否 大 于 0 
} 
return true; 
} 
return false; 


} 

在 isInt 方 法 的 定义 中 ， 需 要 注意 以 下 两 点 。 

口 第 一 个 参数 前 面 有 个 this 关键 字 ， 表 示 这 是 一 个 扩展 方法 。 

口 this 关键 字 后 面 是 string 类 型 ， 表 示 这 个 扩展 方法 是 对 string 类 的 扩展 。 

到 此 为 止 ， 已 经 在 string 类 中 添加 了 一 个 重 载 的 isInt 方法， 可 以 在 代码 中 使 用 了 。 而 
且 这 个 isInt 方 法 可 以 被 Visual Studio 开发 环境 所 识别 ， 如 图 8.2 所 示 。 


[inGtrinet] we 


图 8.2 Visual Studio 自动 识别 扩展 方法 
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(4) 在 Main0 方 法 中 ， 输 入 以 下 代码 ， 测 试 ismnt 扩展 方法 。 
static void Main(string[] args) 
{ 
string text = "=-9123"; 
bool b = text.isInt(); 
Console.WriteLine (text + (b ? "" : "不 ") + "是 整数 "); 
b = text.isInt (true); 
Console.WriteLine (text + (b ? "" : "不 ") + "是 正 整 数 "); 
// 由 于 引号 中 的 内 容 是 string 类 型 ， 所 以 以 下 代码 也 合法 
b= “T0013Int() 汉 
3 
(5) 运行 此 程序 ， 输 出 结果 如 下 : 


-9123 是 整数 
-9123 不 是 正 整 数 


8.1.5 Lambda 表达 式 


Lambda 表达 式 是 对 匿名 方法 的 一 种 改进 ， 具 有 更 加 简洁 的 语法 和 更 易 理 解 的 形式 。 
Lambda 表达 式 可 以 包含 表达 式 和 语句 ， 并 且 可 用 于 创建 委托 或 表达 式 目录 树 类 型 。 所 有 
Lambda 表达 式 都 使 用 Lambda 运算 符 =>， 该 运算 符 读 为 “goes to”。 该 Lambda 运算 符 的 
左边 是 输入 参数 ， 右 边 包含 表达 式 或 语句 块 。 声 明 Lambda 表示 式 的 语法 如 下 : 

(参数 列表 ) => {方法 体 } 


其 中 的 参数 列表 与 普通 方法 的 参数 列表 相同 。Lambda 表达 式 是 升级 版 的 匿名 方法 ， 
其 语法 形式 与 匿名 方法 以 及 普通 方法 的 语法 都 有 很 大 的 相似 之 处 ， 如 下 面 的 语句 所 示 。 


返回 类 型 方法 名 (参数 列表 ) { 方法 体 } // 普 通 方法 
delegate (参数 列表 ) { 方法 体 } // 匿 名 方法 
(参数 列表 ) => { 方法 体 } //Lambda 表达 式 


上 述 语 法 描述 中 ， 第 1 行 是 普通 的 方法 声明 ， 第 2 行 是 匿名 方法 ， 第 3 行 是 Lambda 
表达 式 。 通 过 三 者 的 对 比 看 出 ，Lambda 表达 式 在 语法 形式 上 相当 于 把 普通 方法 声明 中 的 
返回 类 型 省 略 , 并 且 在 参数 列表 和 方法 体 之 间 加 了 一 个 Lambda 操作 符 =>。Lambda 表达 式 
的 返回 类 型 由 语句 块 中 的 retum 语句 所 决定 。 下 面 的 代码 是 几 个 Lambda 表达 式 的 例子 。 

// 求 两 个 整数 最 大 值 

(int x, int y) => { return x>y?x:y; } 

// 决 断 一 个 字符 串 是 否 数字 

(string text) => 

{ 


double d; 
return double.TryParse (text, out d); 
} 
// 输 入 Hello, world 
() => { Console.WriteLine ("Hello，World!"); } 


和 注意 : 如 果 Lambda 表达 式 没有 参数 ， 则 参数 列表 为 空 ， 但 必须 要 有 圆 括号 。 
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如 果 Lambda 表达 式 只 能 使 用 上 述 语法 形式 ， 那 么 Lambda 表达 式 和 匿名 方法 也 就 没 
有 多 大 区 别 。 正 是 由 于 Lambda 表达 式 允 许 在 特定 情况 下 对 某 些 元 素 省 略 ， 使 Lambda 表 
达 式 更 加 灵活 易 读 。Lambda 表达 式 具 有 以 下 省 略语 法 。 

(1) 如 果 Lambda 表达 式 的 参数 类 型 可 以 通过 上 下 文 推断 时 ， 参 数列 表 中 的 类 型 名 称 
可 以 被 省 略 。 下 面 的 代码 是 一 个 省 略 了 参数 类 型 的 Lambda 表达 式 。 


(x,y)=>{return x>y?x:y;} 


(2) 如 果 Lambda 表达 式 只 有 一 个 参数 且 参 数 类 型 被 省 略 ， 则 参数 列表 外 面 的 
也 可 以 省 略 ， 如 下 代码 所 示 。 


x=>{return X++7 


(3) 如 果 Lambda 语句 的 方法 体 只 有 一 条 retum 语句 ， 且 retum 语句 有 返回 类 型 ， 则 
TIetum 关键 字 、 分 号 、 大 括号 都 可 以 省 略 ， 此 时 的 Lambda 表达 式 的 方法 体 只 剩 下 一 个 表达 
式 。 下 面 是 两 个 例子 。 

(x,y)=>x>y?x:y 

// 上 述 代码 相当 于 (x,y)=>{return x>y?x:y;} 

X=>X++ 

// 上 面 一 行 代码 相当 于 x=>{return x++;} 

Lambda 表达 式 用 的 最 多 的 地 方 就 是 集合 操作 ， 如 元 素 查找 。 下 面 给 出 一 个 集合 操作 
的 例子 ， 说 明 Lambda 表达 式 的 应 用 。 

【 例 8-3】 Lambda 表达 式 应 用 。 

本 例 创建 一 个 学 生成 绩 的 列表 ， 并 利用 Lambda 表达 式 作 为 查询 条 件 ， 对 学 生成 绩 列 
表 进 行 查找 、 删 除 等 操作 。 


class Program 


{ 


a 
en 


static void Main(string[] args) 
List<int> scores = new List<int>(); // 学 生成 绩 列 表 
// 随 机 填充 学 生成 绩 
Random rand = new Random(); 
EOE (nt = On i < oD Lr 

scores.Add (rand.Next (20, 100)); 

Console.WriteLine ("原始 数据 "); 
printScores (scores); 
// 查 找 90 分 以 上 的 人 
int match = scores.Count (n => n>=90); 
Console.WriteLine (match+" 人 成 绩 高 于 90 分 "); 
// 删 除 不 及 格 的 学 生成 绩 
Scores.RemoveAl1 (n => n<60); 
Console.WriteLine ("删除 不 及 格 成 绩 后 的 数据 ") ; 
printScores (scores); 
// 按 照 从 高 到 低 的 顺序 排列 学 生成 绩 
scores.Sort((x, y) => y - x); 
Console .WriteLine ("排序 后 的 数据 ") ; 


PrintScores (scores); 


} 
// 输 出 学 生成 绩 ， 每 行 15 个 


static void printScores (List<int> scores) 


Es 
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Fo (int T= 0F < SecoresCount Ltrt+) 


| 
if (i>0 g&& i $$ 15 == 0) 
Console.WriteLine(); 
Console.Write(string.Format ("{0,—-4}",scores[i])); 
} 


Console.WriteLine(); 
} 
上 述 代 码 运行 结果 如 下 : 


原始 数据 

30 53 314 72 42900436 2933 8913 22202 77 
91 92 37 30 55 58 79 23 64 68 46 50 44 43 37 
40 24 49 47 40 68 40 48 53 51 96 49 33 69 58 
G0 77 717120 36 

5 人 成 绩 高 于 90 分 


删除 不 及 格 成 绩 后 的 数据 
80 91 72 94 89 91 92 79 64 68 68 96 69 60 77 


排序 后 的 数据 
96 94 92 91 91 89 80 79 77 72 69 68 68 64 60 


8.1.6 ”表达 式 树 


表达 式 树 (Expression Tree) 以 数据 形式 表示 语言 级 别 代码 ， 数 据 存储 在 树 形 结构 中 。 
表达 式 目录 树 中 的 每 个 节点 都 表示 一 个 表达 式 , 例 如 一 个 方法 调用 或 诸如 x<y 的 二 元 运算 。 
System.Linq.Expressions.Expression 类 表示 一 个 表达 式 ，Expression 类 是 一 个 抽象 类 ， 可 提 
供用 于 对 表达 式 目录 树 进 行 建 模 的 类 层次 结构 的 根 目 录 。 从 Expression 派生 的 
System.Linq.Expressions 命名 空间 中 的 类 (例如 MemberExpression 和 ParameterExpression 
等 ) 用 于 表示 表达 式 目 录 树 中 的 节点 。Expression 类 包含 一 系列 static 工厂 方法 ， 可 创建 各 
种 类 型 的 表达 式 目录 树 节点 。Expression 主要 有 以 下 派生 类 。 

口 BinaryExpression: 表示 包含 二 元 运算 符 的 表达 式 。 

口 ConditionalExpression: 表示 包含 条 件 运算 符 的 表达 式 。 

口 ConstantExpression: 表示 具有 常量 值 的 表达 式 。 

口 Expression<TDelegate>: 以 表达 式 目 录 树 的 形式 将 强 类 型 Lambda 表达 式 表示 为 数 


据 结构 。 

口 InvocationExpression: 表示 将 委托 或 Lambda 表达 式 应 用 于 参数 表达 式 列表 的 表 
达 式 。 

口 LambdaExpression: 描述 一 个 Lambda 表达 式 。 

口 ListInitExpression: 表示 包含 集合 初始 值 设 定 项 的 构造 函数 调用 。 

口 MemberExpression: 表示 访问 字段 或 属性 。 

口 MemberInitExpression: 表示 调用 构造 函数 并 初始 化 新 对 象 的 一 个 或 多 个 成 员 。 

口 MethodCallExpression: 表示 调用 一 种 方法 。 

口 NewArrayExpression: 表示 创建 新 数组 并 可 能 初始 化 该 新 数组 的 元 素 。 

口 NewExpression: 表示 构造 函数 调用 。 
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口 ParameterExpression: 表示 命名 的 参数 表达 式 。 

口 TypeBinaryExpression: 表示 表达 式 和 类 型 之 间 的 操作 。 

口 UnaryExpression: 表示 包含 一 元 运算 符 的 表达 式 。 

下 面 通过 图 形 的 方式 形象 地 表示 表达 式 树 的 结构 。 平 面 几何 中 求 扇形 面积 的 公式 为 : 

Area 一 angle / 360 * 3.14 * radius * radius, 其 中 angle 表示 扇形 圆心 角 , radius 表示 扇形 
半径 。 将 这 个 公式 化 简 可 以 得 到 : 

Area = angle * (3.14/ 360) * radius * radius= ( angle * 0.0087 ) * ( radius * radius ) 

将 上 述 公 式 用 Lambda 表达 式 表 示 ， 可 得 

Expression<Func<double, double, double>> areaExpression = (r, a) => (a * 


OO ww Cr es 


上 述 表 达 式 所 对 应 的 表达 式 树 图 形 如 图 8.3 所 示 。 


(r, a)=a*0.087*r*r Lambda 表达 式 


表达 式 树 


Expression<Func<double, double, double>> | 


a 
参数 


BinaryExpression 


(Multiply) 
2 A 
BinaryExpression BinaryExpression 
(Multiply) (Multiply) 
= 操作 左 操作 数 右 操作 数 
左 操 作 数 Re 1 探 作 
和 右 操作 数 
| 
ParameterExpression ParameterExpression ConstantExpression 

(a) (n) (0.087) 

4 


图 8.3 扇形 面积 表达 式 树 


可 以 用 以 下 C# 代 码 构造 如 图 8.3 所 示 的 表达 式 树 。 
// 构 建 两 个 参数 表达 式 (半径 < 和 角度 a) 


ParameterExpression radiusParam=Expression.Parameter (typeof (double), 

mn) 

ParameterExpression angleParam = Expression.Parameter (typeof (double), 
wan); 

// 构 建 常量 表达 式 (3 .14/360=0.0087) 

ConstantExpression constExp = Expression.Constant (0.0087, typeof (double)); 
// 构 建 两 个 乘法 表达 式 a*0.0087 和 r*r 

BinaryExpression mull = Expression.Multiply (angleParam, constExp); 
BinaryExpression mul2 = Expression.Multiply (radiusParam, radiusParam); 


// 将 上 面 两 个 乘法 表达 式 相 乘 ， 得 到 面积 计算 表达 式 
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BinaryExpression result = Expression.Multiply (mull, mul2); 


// 将 面积 表达 式 封装 为 Lambda 表达 式 

Expression<Func<double, double, double>> areaExpression = 
Expression.Lambda<Func<double, double, double>> 
(result, new ParameterExpression[] { radiusParam,angleParam }); 


8.2 LINQ 基本 操作 


LINQ 提供 了 一 种 跨 各 种 数据 源 和 数据 格式 使 用 数据 的 一 致 模型 。 为 了 实现 这 种 一 致 
性 ，LINQ 充分 利用 C# 中 的 扩展 方法 、Lambda 表达 式 、 对 象 初始 化 器 、 匿 名 类 型 等 特性 。 
LINQ 为 IEnumerable<T> 泛 型 接口 添加 了 若干 扩展 方法 ，NET 中 所 有 泛 型 集合 类 都 实现 了 
IEnumerable 接口 ， 通 过 这 些 扩展 方法 可 以 实现 对 泛 型 集合 的 查询 功能 。 

LINQ 的 查询 有 两 种 语法 ， 一 种 是 方法 语法 ， 即 使 用 传统 的 C# 类 和 方法 的 语法 进行 查 
询 , 另外 一 种 语法 是 查询 语法 , 这 是 为 LINQ 而 引入 的 新 的 C# 语 法 。 这 两 类 语法 形式 不 同 ， 
但 是 功能 相同 。 实 际 上 C# 在 遇 到 LINQ 的 查询 语法 时 ， 会 翻译 成 相应 的 方法 语法 再 执行 。 


8.2.1 创建 查询 数据 源 


LINQ 用 于 从 各 种 数据 源 中 查询 数据 。 为 了 演示 LINQ 的 语法 和 功能 ， 需 要 创建 一 部 
分 数据 ， 以 这 些 数 据 为 基础 进行 查询 。 本 节 将 使 用 一 组 随机 生成 的 商品 销售 数据 为 数据 源 
演示 LINQ 的 使 用 。 与 此 相关 的 有 3 个 类 : 商品 类 别 Category、 商 品 信 息 Product 和 销售 记 
录 ProductSale。 为 了 随机 生成 数据 ，3 个 类 中 分 别 包含 一 个 随机 填充 数据 的 方法 。Category 
类 的 代码 如 下 : 
public class Category 
{ 
public string id { get; set; } // 类 别 ID 
public string name { get; set; } // 类 别名 称 
/// <summary> 
/// 随机 生成 商品 类 别 
/// </summary> 
/// <param name="count"> 要 生成 的 商品 类 别 数 量 </param> 
/// <returns> 生 成 的 商品 类 别 列表 </returns> 


public static List<Category> randomCategories (int count) 


Random r=new Random(DateTime.Now.Millisecond); 

List<Category> categories = new List<Category>(); 

// 循 环 生成 类 别 数据 ， 不 保证 类 别 ID 唯一 

for (int i = 0; i < count; i++) 

{ 
Category category = new Category(); 
category.id = "C" + r.Next(1000); 
category.name = "Category " + i.ToString(); 
categories.Add (category); 

站 


return categories; 
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i 


Product 类 的 代码 如 下 : 

public class Product 
public string id { get; set; } // 商 品 ID 
public string name { get; set; } // 名 称 
public int storage { get; set; } // 库 存 
public double price { get; set; } // 价 格 
public Category category { get; set; } // 类 别 


有 


/// <summary> 
/// 随机 生成 商品 列表 
/// </summary> 
/// <param name="count"> 要 生成 的 商品 数量 </param> 
/// <returns> 所 生成 的 商品 列表 </returns> 
public static List<Product> randomProducts (int count) 
Random r = new Random(DateTime.Now.Millisecond); 
List<Product> products = new List<Product>(); 
// 随 机 生成 类 别 ， 设 类 别 数量 为 商品 数量 的 1/10， 但 至 少 有 3 种 ， 最 多 有 20 种 
int categoryCount = Math.Min(20, Math.Max(3, count / 10)); 
Category[] categories = Category.randomCategories (categoryCount). 
ToArray (); 
for (int i = 0; i < count; i++) 
{ 
Product p = new Product(); 
.id = "P" + r.Next(10000) .ToString("0000") 7 
.name = "Product " + i.ToString(); 
-Storage = r.Next (10, 10000); 
-Price = 1 + r.NextDouble() * 300; 
.Category = categories[r.Next (categoryCount)]; 
products.Add (p); 


ee ee Bone) 


} 


return products; 


ProductSale 类 的 代码 如 下 : 
public class ProductSale 


{ 


. 294 . 


public Product product { get; set; } // 所 销售 商品 
public DateTime date { get; set; } // 销 售 日 期 
public int quantity { get; set; } // 销 售 数量 


/// <summary> 
/// 随机 生成 一 组 销售 数据 
/// </summary> 
/// <param name="count"> 要 生成 的 销售 记录 的 数量 </param> 
/// <returns> 所 生成 的 销售 数据 列表 </returns> 
public static List<ProductSale> randomSales (int count) 
1 
// 随 机 生成 商品 列表 ， 设 商品 种 类 为 销售 记录 的 1/5 
int productCount = count / 5 +1; 
Product [] products = Product.randomProducts (productCount). 
ToArray (); 
Random r = new Random(DateTime.Now.Millisecond); 
List<ProductSale> sales = new List<ProductSale>(); 
For (Int i = ONE < Conts TE) 


{ 
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ProductSale sale = new ProductSale () 7 
sale.product = products[r.Next (productCount)]; 
sale.quantity = r.Next (100); 
sale.date = DateTime.Now.AddDays (r.Next (-30, 30)); 
sales.Add (sale); 

3 


return sales; 


8.2.2 投影 


在 数据 查询 中 投影 是 指 从 一 个 数据 源 中 查询 某 些 字段 或 属性 ， 例 如 SQL 语句 中 的 
“select 列 名 1, 列 名 2, …, 列 名 N from 表 名 ”就 是 一 个 投影 操作 。 在 LINQ 中 ,IEnumerable<T> 
类 的 Select 扩展 方法 可 以 实现 投影 操作 ， 方 法 声明 如 下 : 


// 摘 要 :将 序列 中 的 每 个 元 素 投影 到 新 表 中 

// 参 数 : 

//source: 一 个 值 序列 ， 要 对 该 序列 调用 转换 函数 

//selector: 应 用 于 每 个 元 素 的 转换 函数 

// 类 型 参数 : 

//TSource: source 中 的 元 素 的 类 型 

//TResult: selector 返回 的 值 的 类 型 

// 返 回 结果 : 一 个 System.Collections .Generic.IEnumerable<T>, 其 元 素 为 对 source 
的 每 个 元 素 调用 转换 函数 的 结果 

// 异 常 :System.ArgumentNullException: source 或 selector 为 null 

public static IEnumerable<TResult> Select<TSource, TResult> (this 
IEnumerable<TSource> source, Func<TSource, TResult> selector); 


Select 方法 用 于 从 集合 中 查询 数据 ， 所 查询 到 的 数据 并 不 一 定 与 集合 中 原 有 数据 类 型 
相同 ， 而 是 可 以 基于 原 有 数据 创建 任意 的 数据 类 型 。 例 如 ， 可 以 从 一 个 学 生 信息 集合 中 查 
询 所 有 的 学 号 。Select 方法 参数 中 的 Func<TSource, TResult> 实 现 了 这 个 类 型 转换 功能 ， 通 
常 为 这 个 参数 传递 一 个 Lambda 表达 式 。 

【 例 8-4】 Select 查询 。 

本 例 演示 如 何 使 用 IEumerable<T> 类 的 Select 扩展 方法 查询 商品 信息 。 


public void selectDemo () 

{ 
List<Product> products = Product.randomProducts (5); 
var query = products.Select (p => p); // 查 询 商 品 本 身 
foreach (var item in query) 


{ 


Console .WriteLine ("Id={0},Name={1},Price={2}, storage={3},category= 
Es 
item.id, item.name, item.price, item.storage, item.category. 
name); 
L 
// 查 询 商品 的 一 部 分 信息 ， 这 些 信息 构成 一 个 匿名 类 型 
Var query2 = products.Select (p => new { Id=p.-id，Name =p.name, Category 
= p.category.name }); 
foreach (var item in query2) 


上 
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| 


Console.Write("Id={0},Name={1},Category={2}", item.Id, item.Name, 
item.Category); 


上 述 代 码 运 行 结果 如 下 : 


Id=P3849,Name=Product 

0,Price=123.874712721852, storage=7362,category=Category 0 
Id=P7290,Name=Product 1,Price=263.713322072622, storage=4351, category= 
Category 2 

Id=P0212,Name=Product 2,Price=10.2506326778096, storage=3186,category= 
Category 0 

Id=P0479,Name=Product 3,Price=21.8219276838107, storage=1460, category= 
Category 2 

Id=P2850,Name=Product 4,Price=283.019849439161, storage=8551, category= 
Category 2 

Id=P3849,Name=Product 0,Category=Category 0 

Id=P7290,Name=Product 1,Category=Category 2 

Id=P0212,Name=Product 2,Category=Category 0 

Id=P0479,Name=Product 3,Category=Category 2 

Id=P2850,Name=Product 4,Category=Category 2 


如 前 所 述 ，LINQ 有 两 种 语法 : 方法 语法 和 查询 语法 。LINQ 的 查询 语法 如 下 : 


from 变量 名 in 数据 源 
select 表达 式 


例 8-4 所 使 用 的 语法 为 扩展 方法 语法 ， 所 对 应 的 等 价 查询 语法 如 下 : 


public void selectDemo () 


有 


8.2.3 


List<Product> products = Product.randomProducts (5); // 随 机 生成 商品 列表 
var query = from p in products 

select p; 
foreach (var item in query) 


! 


Console.WriteLine ("Id={0},Name={1},Price={2}, storage={3},category={4 
er 
item.id, item.name, item.price, item.storage, item.category. 
name); 
} 
// 查 询 商品 的 一 部 分 信息 ， 这 些 信息 构成 一 个 匿名 类 型 
Var query2 = from p in products 
select new { Id=p.id, Name =p.name, Category =p.category. 
name }; 
foreach (var item in query2) 
Console.WriteLine ("Id={0},Name={1},Category={2}", item.Id, item. 
Name, item.Category); 


数据 查询 中 的 选择 是 指 从 数据 源 中 查找 符合 条 件 的 数据 ， 过 滤 掉 不 符合 条 件 的 数据 。 
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SQL 语言 中 Select 语句 后 面 的 Where 条 件 所 起 的 作用 就 是 选择 。 在 LINQ 中 IEnumerable<T> 
类 的 Where 扩展 方法 可 以 实现 数据 过 滤 功 能 ，Where 方法 声明 如 下 : 


// 摘 要 : 基于 谓词 筛选 值 序列 

// 参 数 : 

/V/source: 要 筛选 的 System.Collections .Generic.IEnumerable<T> 
//predicate: 用 于 测试 每 个 元 素 是 否 满足 条 件 的 函数 

// 类 型 参数 : 

//TSource: source 中 的 元 素 的 类 型 

// 返 回 结果 : 

// 一 个 System.Collections.Generic.IEnumerable<T>, 包含 输入 序列 中 满足 条 件 的 元 素 
// 异 常 : System.ArgumentNullException: source 或 predicate 为 null 
public static IEnumerable<TSource> Where<TSource> (this IEnumerable 
<TSource> source, Func<TSource, bool> predicate); 


与 Where 扩展 方法 相对 应 的 查询 语法 为 : 


from 变量 名 in 数据 源 
where 条 件 
select 表达 式 


【 例 8-$】 Where 过 滤 数 据 。 
本 例 演 示 使 用 Where 方法 查询 价格 大 于 180 元 的 商品 信息 ， 代 码 中 同时 给 出 了 方法 语 
法 和 查询 语法 两 种 形式 ， 其 功能 是 等 价 的 。 


public void whereDemo () 


List<Product> products = Product.randomProducts (10);// 随 机 生成 商品 列表 
var query = from p in products 
where p.price > 180 
select p; //LINQ 查询 
// 或 者 使 用 查询 方法 : var query=products.Where(p => p.price > 180); 
foreach (var item in query) 
Console.WriteLine ("Id={0},Name={1},Price={2}", 
item.id, item.name, item.price); 
} 
var query2 = from p in products 
where p.price > 180 
select new { Id = p.id, Name = p.name, Category = p.cate— 
gory.name }; 
// 或 者 使 用 查询 方法 ， 如 下 代码 所 示 
//var query2 = products.Where(p => p.price > 180) 
KK -Select(P => new { Id = p.id, Name = p.name, Category = p. 
category.name }); 
foreach (var item in query2) 
Console.WriteLine ("Id={0},Name={1},Category={2}", item.Id, item. 
Name, item.Category); 


; 
上 述 代码 运行 结果 如 下 : 
Id=P1569,Name=Product 1,Price=199.694056784126 


Id=P1562,Name=Product 2,Price=240.83325019471 
Id=P6013,Name=Product 3,Price=290.243594738303 
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Id=P1739,Name=Product 9,Price=284.309072900242 
Id=P1569,Name=Product 1,Category=Category 1 
Id=P1562,Name=Product 2,Category=Category 0 
Id=P6013,Name=Product 3,Category=Category 2 
Id=P1739,Name=Product 9,Category=Category 1 


8.2.4 排序 


为 了 实现 数据 排序 ，IEnumerable<T> 类 包含 4 个 相关 的 扩展 方法 ， 分 别 为 OrderBy 方 
法 、OrderByDescending 方法 、ThenBy 方法 、ThenByDescending 方法 。 其 中 OrderBy 方法 
的 功能 是 将 数据 源 按照 一 个 表达 式 进行 升序 排序 , OrderByDescending 方法 的 作用 是 按照 一 
个 表达 式 进行 降序 排序 .ThenBy 方法 的 作用 是 对 一 个 已 经 按照 某 表 达 式 排序 的 集合 再 按照 
另 一 个 表达 式 进行 排序 ，henByDescending 方法 的 功能 与 之 类 似 ， 只 是 按照 降序 排序 。 
OrderBy 方法 声明 如 下 代码 所 示 ， 其 他 3 个 方法 声明 与 之 类 似 。 


// 摘要 :根据 键 按 升序 对 序列 的 元 素 排序 

// 参数 : 

// source: 一 个 要 排序 的 值 序列 

// keySelector: 用 于 从 元 素 中 提取 键 的 函数 

// 类 型 参数 : 

// TSource: source 中 的 元 素 的 类 型 

// TKey:keySelector 返回 的 键 的 类 型 

// 返回 结果 :一 个 System.Linq.IOrderedEnumerable<TElement>， 其 元 素 按 键 排序 
// 异常 : System.ArgumentNullException: source 或 keySelector 为 null 
public static IOrderedEnumerable<TSource> OrderBy<TSource, TKey> 
(this IEnumerable<TSource> source, Func<TSource, TKey> keySelector); 


与 OrderBy 等 4 个 排序 方法 相对 应 的 查询 语法 如 下 : 


from 变量 名 in 数据 源 

where 条 件 

orderby 表达 式 1 [descending] ,表达 式 2 [descending] ，… ,表达 式 n [descending] 
select 表达 式 


【 例 8-6】 数据 排序 。 
本 例 演示 按照 单 关 键 字 和 多 关键 字 对 商品 信息 进行 排序 。 
public void sortDemo () 
{ 
List<Product> products = Product.randomProducts (10); 
// 把 商品 按照 价格 排序 升序 》 
Var query = from p in products 
orderby p.price 
where p.price>180 
select p; 
foreach (var item in query) 
{ 
Console.WriteLine ("Id={0},Name={1},Price={2}", 
item.id, item.name, item.price); 
} 
// 上 述 语 句 也 可 以 采用 查询 方法 写成 以 下 形式 
//var query=products.OrderBy(p => p.price); 
// 把 商品 先 按 照 类 别 编号 (升序 ) 和 价格 (降序) 排序 


Var query2 = from p in products 
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where p.price<100 
orderby p.category.id,p.price descending 
select p; 
// 上 述 语 名 也 可 以 采用 查询 方法 写成 以 下 形式 
//var query2 = products .OrderBy (p => p.category.id) .ThenByDescending (p 
=> p.price); 
foreach (var item in query2) 


下 
Console.WriteLine ("Id={0},Name={1},Price={2}", 
item.id, item.name, item.price); 
} 
} 
8.2.5 数据 分 页 


在 查询 结果 包含 大 量 数 据 时 ， 出 于 性 能 方面 的 考虑 ， 通 常 需 要 将 查询 结果 分 页 显示 。 
IEnumerable<T> 类 的 Skip 扩展 方法 和 Take 扩展 方法 很 方便 地 能 够 实现 数据 分 页 功能 。Take 
方法 的 作用 是 从 序列 的 开头 返回 指定 数量 的 连续 元 素 ，Skip 方法 的 作用 是 跳 过 序列 中 指定 
数量 的 元 素 ， 然 后 返回 剩余 的 元 素 ， 这 两 个 方法 配合 使 用 ， 先 跳 过 一 定数 量 的 元 素 ， 再 取 
一 定数 量 的 元 素 ， 就 实现 了 分 页 功能 。Skip 方法 和 Take 方法 声明 如 下 。 


// 摘 要 : 跳 过 序列 中 指定 数量 的 元 素 ， 然 后 返回 剩余 的 元 素 
// 参 数 : 
//source: 要 从 中 返回 元 素 的 System.Collections.Generic.IEnumerable<T> 
//count: 返回 剩余 元 素 前 要 跳 过 的 元 素数 量 

// 返 回 结果 : 

// 一 个 System.Collections.Generic.IEnumerable<T>,， 包含 输入 序列 中 指定 索引 后 出 现 
的 元 素 

// 异 常 : System.ArgumentNullException: source 为 null 

public static IEnumerable<TSource> Skip<TSource> (this IEnumerable<TSource> 
source, int count); 

// 摘 要 : 从 序列 的 开头 返回 指定 数量 的 连续 元 素 

// 参 数 : 

//source: 要 从 其 返回 元 素 的 序列 

//count: 要 返回 的 元 素数 量 

// 返 回 结果 : 

// 一 个 System.Collections.Generic.IEnumerable<T>, 包含 输入 序列 开头 的 指定 数量 的 
元 素 

// 异 常 : System.ArgumentNullException: source 为 null。 

public static IEnumerable<TSource> Take<TSource> (this IEnumerable<TSource> 
source, int count); 


【 例 8-7】 数据 分 页 。 
本 例 演示 LINQ 的 数据 分 页 功能 ， 查 找 所 有 商品 中 单价 大 于 200 元 的 商品 ， 把 查询 结 
果 按 价格 降序 排列 ， 并 分 页 输出 。 


public void skipTake () 
{ 


List<Product> products = Product.randomProducts (100); 
// 查 找 商 品 列表 中 单价 高 于 200 的 商品 ， 按 照 价格 降序 排列 
Var query = from p in products 

where p.price > 200 

orderby p.price descending 
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select p; 

// 设 置 页 面 大 小 为 3， 并 计算 总 页 数 
int pageSize = 3; 
int pageCount = (int)Math.Ceiling (query.Count() *1.0 / pageSize); 
/ /循环 输出 每 页 数据 
for (int i = 0; i < pageCount; i++) 
| 

Console .WriteLine ("= 一 = 第 {0} 页 数据 = 一 =", i+1); 

var query2 = query.Skip (i*pageSize) .Take (pageSize); 

foreach (var item in query2) 


{ 
Console.WriteLine ("Id={0},Name={1},Price={2}", 
item.id, item.name, item.price); 


. 
上 述 代 码 部 分 输出 结果 如 下 : 


-一 -第 1 页 数据 =----- 

Id=P2721,Name=Product 85,Price=298.89590802877 
Id=P5429,Name=Product 35,Price=294.308188157765 
Id=P3227,Name=Product 56,Price=286.606975707042 


-=----- 第 2 页 数据 = 

Id=P6665,Name=Product 70,Price=281.919481292795 
Id=P2705,Name=Product 72,Price=276.074729218648 
Id=P7969,Name=Product 22,Price=275.414076830453 


8.2.6 ”数据 分 组 


进行 数据 查询 时 很 多 时 候 需 要 对 数据 进行 分 组 ，IEnumerable<T> 类 的 GroupBy 方法 实 
现 了 分 组 功能 。GroupBy 方法 声明 如 下 。 

// 摘 要 : 根据 指定 的 键 选择 器 函数 对 序列 中 的 元 素 进行 分 组 

// 参 数 : 

/Vsource: 要 对 其 元 素 进行 分 组 的 System.Collections.Generic.IEnumerable<T> 

//keySelector: 用 于 提取 每 个 元 素 的 键 的 函数 

// 类 型 参数 : 

//TSource:source 中 的 元 素 的 类 型 

//TKey:keySelector 返回 的 键 的 类 型 

// 返 回 结果 : 

//IEnumerable<IGrouping<TKey，TSource>>， 其 中 每 个 System.Linq.IGrouping 

<TKey, TElement> 对 象 都 包含 一 个 对 象 序列 和 一 个 键 


// 异 常 :System.ArgumentNul1Exception: source 或 keySelector 为 null 
public static IEnumerable<IGrouping<TKey, TSource>> GroupBy<TSource, 
TKey> (this IEnumerable<TSource> source, Func<TSource, TKey> keySelector); 


【 例 8-8】 分 组 查询 。 
本 例 演示 分 组 查询 功能 ， 将 所 有 商品 按照 类 别 ID 进行 分 组 ， 并 输出 各 组 的 键 值 〈 类 
别 卫 ) 和 各 组 中 的 商品 列表 。 


public void groupDemo () 
{ 
List<Product> products = Product.randomProducts (6) ; 


// 将 商品 按照 类 别 id 分 组 
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Var query = from p in products 
group P by p-.category.id 
into categoryGroup 
select new { categoryid = categoryGroup.Key, products 
= categoryGroup }; 
nb ul 
/ /循环 输 出 每 组 的 类 别 ID 和 组 中 的 商品 


foreach (var group in query) 


1 
Console.WriteLine("======== 第 {0j} 组 (类 别 19:11})= 二 = 一 一 =="， 14+, 
group.categoryid); 
foreach (var item in group.products) 
{ 
Console.WriteLine ("Id={0},Name={1},Price={2}", 
item.id, item.name, item.price); 
} 
} 


上 述 代码 输出 结果 如 下 : 


= ES La d==== 

Id=P2245,Name=Product 0,Price=244.044761169257 
Id=P6378,Name=Product 1,Price=241.861566942586 
Id=P4383,Name=Product 4,Price=271.048425844893 


Id=P8750,Name=Product 3,Price=54.3286917737353 
Id=P5447,Name=Product 5,Price=108.005137254952 


8.2.7 返回 单个 元 素 


根据 主键 的 值 进行 查询 时 ,查询 结果 最 多 只 有 一 项 , 如 下 面 的 SQL 语句 至 多 返回 一 条 
记录 (StudentId 为 主键): 
Select * from Student where StudentId="'BZXY10110201"' 


对 于 这 种 最 多 只 返回 一 条 数据 的 查询 ， 在 LINQ 中 可 以 使 用 IEnumerable<T> 的 Single 
和 SingleOrDefault 扩展 方法 。Single 方法 用 于 读 取 不 多 不 少 单条 记录 ， 如 果 查 询 结果 多 于 
或 者 少 于 一 条 记录 都 会 引发 异常 。SingleOrDefault 返回 数据 源 中 的 单条 记录 或 者 返回 空 ( 如 
果 数 据 源 为 空 )， 如 果 数 据 源 包含 多 条 数据 ， 则 引发 异常 。Single 和 SingleOrDefault 方法 声 
明 如 下 。 


// 摘 要 : 返回 序列 的 唯一 元 素 ; 如 果 该 序列 并 非 恰 好 包含 一 个 元 素 ， 则 会 引发 异常 

// 参 数 : 

//source: 一 个 System.Collections .Generic.IEnumerable<T>， 用 于 返回 单个 元 素 
// 类 型 参数 : 

//TSource: source 中 的 元 素 的 类 型 

// 返 回 结果 : 输入 序列 的 单个 元 素 

// 异 常 : 

//System.ArgumentNullException: source 为 null 
//System.InvalidOperationException: 输入 序列 包含 多 个 元 素 。- 或 -输入 序列 为 空 


public static TSource Single<TSource> (this IEnumerable<TSource> source); 


// 摘 要 : 如 果 该 序列 包含 多 个 元 素 ， 此 方法 将 引发 异常 
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// 返 回 结果 : 返回 输入 序列 的 单个 元 素 ; 如 果 序 列 不 包含 任何 元 素 ， 则 返回 default (TSource) 
// 异常 : 

//System.ArgumentNullException: source 为 null 
//System.InvalidOoperationException: 输入 序列 包含 多 个 元 素 

public static TSource SingleOrDefault<TSource> (this IEnumerable<TSource> 
source); 


对 于 返回 多 条 记录 的 查询 ， 如 果 想 只 取 一 条 记录 ， 则 可 以 使 用 IEnumerable<T> 的 First 
和 FirstOrDefault 扩展 方法 。 Fist 方法 的 作用 是 返回 查询 结果 的 首 元 素 , 如 果 查 询 结 果 为 空 ， 
则 引发 异常 。FirstOrDefault 方法 的 作用 是 返回 查询 结果 的 首 元 素 ， 如 果 查 询 结果 为 空 ， 则 
返回 元 素 类 型 的 默认 值 (通常 为 null)。First 和 FirstOrDefault 方法 声明 如 下 : 


// 摘 要 :返回 序列 中 的 第 一 个 元 素 

// 参 数 : 

//source: 要 返回 其 第 一 个 元 素 的 System.-Collections .Generic.IEnumerable<T> 

// 类 型 参数 : TSource: source 中 的 元 素 的 类 型 

// 返 回 结果 : ”返回 指定 序列 中 的 第 一 个 元 素 

// 异 常 : 

//System.ArgumentNullException: source 为 null 
//System.InvalidoperationException: 源 序列 为 空 

public static TSource First<TSource> (this IEnumerable<TSource> source); 
// 摘 要 : 返回 序列 中 的 第 一 个 元 素 ; 如 果 序 列 中 不 包含 任何 元 素 ， 则 返回 默认 值 

// 返 回 结果 : 

// 如 果 source 为 室 ， 则 返回 default (TSource) ; 否则 返回 source 中 的 第 一 个 元 素 

// 异 常 : System.ArgumentNullException: source 为 null 

public static TSource FirstOrDefault<TSource>(this IEnumerable<TSource> 
source); 


【 例 8-9】 查询 单个 元 素 。 
本 例 演示 使 用 Single、SingleOrDefault、First、FirstOrDefault 方法 查询 单个 元 素 。 首 先 
生成 一 个 商品 列表 ， 然 后 分 别 调用 以 上 4 个 方法 查询 商品 列表 中 价格 大 于 200 元 的 单个 商品 。 


public void singleFirst() 


“302" 


List<Product> products = Product.randomProducts (6) ; 
// 查 询 商品 列表 中 单价 大 于 200 的 商品 
var query = from p in products 
where p.price > 200 
select p; 
Product item=null; 
// 调 用 Single 方法 进行 查询 
try 
item = query.Single(); 
Console.WriteLine ("Single 方法 查询 结果 为 : ") ; 
Console.WriteLine("Id={0},Name={1},Price={2}", 
item.id, item.name, item.price); 
} 
catch (InvalidOperationException ex) 
{ 
Console.WriteLine ("Single 发 生 异 常 。 查 询 结果 不 是 正好 包含 单条 数据 。") ; 
Console.WriteLine ("异常 详细 信息 : "+ex.Message); 
l 
// 调 用 SingleOrDefault 方法 进行 查询 
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try 


item = query.SingleOrDefault (); 
Console.WriteLine ("SingleOrDefault 方法 查询 结果 为 : ") ; 
4£° (item == null) 
Console.WriteLine ("<NULL>"); 
else 
Console.WriteLine ("Id={0},Name={1},Price={2}", 
item.id, item.name, item.price); 


catch (InvalidOperationException ex) 


" 


Console.WriteLine ("SingleOrDefault 发 生 异 常 。 查 询 结果 包含 多 于 一 条 数据 。 


Console.WriteLine (" 异 常 详细 信息 : " + ex.Message) ; 
} 
// 调 用 First 方法 进行 查询 
try 


item = query.First (); 

Console.WriteLine ("First 方法 查询 结果 为 :"); 

Console.WriteLine ("Id={0},Name={1},Price={2}", 
item.id, item.name, item.price); 


catch (InvalidOperationException ex) 


Console.WriteLine ("Single 发 生 异常 。 查 询 结 果 不 包含 数据 。") ; 
Console.WriteLine ("异常 详细 信息 : " + ex.Message); 


// 调 用 FirstorDefault () 方 法 进行 查询 
item = query.FirstOrDefault (); 
Console.WriteLine ("FirstOrDefault 方法 查询 结果 为 : ") ; 
if (item == null) 
Console.WriteLine ("<NULL>"); 
else 
Console.WriteLine ("Id={0},Name={1},Price={2}", 
item.id, item.name, item.price); 


8.2.8 延迟 执行 和 立即 执行 


在 LINQ 表达 式 的 fom*…where…select 语句 中 ， 并 没有 执行 实际 的 查询 工作 ， 而 仅 
是 生成 了 一 个 查询 命令 , 实际 的 查询 要 等 到 使 用 查询 结果 时 (例如 在 foreach 语句 中 使 
询 结果 ) 才 被 执行 。 这 种 现象 叫做 LINQ 的 延迟 执行 。 如 果 想 让 查询 立即 执行 ， 可 以 调 
查询 的 ToList 方法 ， 将 查询 结果 转换 为 列表 ， 从 而 强制 查询 立即 执行 。 下 面 通过 一 个 例 
说 明 查 询 的 延迟 执行 和 立即 执行 。 

【 例 8-10】 LINQ 延迟 执行 。 

(1) 创建 一 个 控制 台 应 用 程序 。 

(2) 向 项 目 中 添加 一 个 Person 类 ， 代 码 如 下 : 


class Person 


{ 


查 


习 
] 
子 


-Ms 
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public string name { get; set; } 
public string sex { get; set; } 
public int age { get; set; } 
public override string ToString () 
. 
Console.WriteLine ("Person.ToString() :\t 姓名 "+name); 
/7 线程 休眠 1 秒 钟 
System.Threading.Thread.Sleep(1000); 
return string.Format (" 姓 名 : {0} 性别: {1} 年 龄 : {2}", name, sex,age); 


} 
(3) 在 Main 方法 中 编写 以 下 代码 : 


static void Main(string[] args) 
{ 
List<Person> people = new List<Person>(); 
people.Add (new Person { name = " 张 三 "，sex = " 男 ", age = 23 }); 
people.Add (new Person { name = " 李 四 ", sex = " 男 ", age = 28 }); 
people.Add (new Person { name = "小 妹 "，sex = " 女 "，age = 17 }); 
people.Add (new Person { name = " 赵 芳 "，sex = " 女 "，age = 22 }); 
Var temp = from P in people 
where p.age>20 
select p.ToString(); 
Console.WriteLine ("Main 方法 :\t 年 龄 大 于 20 的 人 : "); 
foreach (var s in temp) 


{ 


Console.WriteLine ("Main 方法 :\t"+s); 
} 

: 

(4) 运行 此 程序 ， 可 以 看 到 ，Person.ToString 方法 和 Main 方法 的 代码 是 交替 执行 的 。 
也 就 是 说 ， 在 Main 方法 的 foreach 语句 中 ， 每 当 使 用 temp 结果 集中 的 一 个 元 素 时 ， 这 个 
元 素 才 被 计算 生成 , 而 不 是 在 声明 temp 结果 集 的 时 候 一 次 性 生成 所 有 元 素 。 程序 输出 结果 
如 下 : 

Main 方 法: ”年龄 大 于 20 的 人 : 

Person.ToString () : 姓名 张 三 

Main 方 法 : ”姓名 : 张 三 性 别 : 男 年 龄 : 23 

Person.ToString() : 姓名 李 四 

Main 方法 : ”姓名 : 李 四 性 别 : 男 年 龄 : 28 

Person.ToString() : 姓名 赵 芳 

Main 方法: ”姓名 : 赵 芳 性 别 : 女 年 龄 : 22 

(5) 由 于 LINQ 查询 的 延迟 执行 ， 所 以 在 读 取 LINQ 查询 结果 集 时 ， 其 中 的 元 素 总 是 
最 新 的 ， 即 使 在 定义 了 LINQ 查询 以 后 ， 源 数据 集合 发 生 了 改变 。 下 面 通过 实际 例子 来 说 
明 这 一 点 。 修 改 Main 方法 的 代码 如 下 : 

static void Main(string[] args) 


List<Person> people = new List<Person>(); 
people.Add (new Person { name = " 张 三 ", sex = " 男 ", age = 23 }); 
people.Add (new Person { name = " 李 四 "，sex = " 男 "，age RE 
people.Add (new Person { name = je 加 抽 sex = bi age = 17 }); 
people.Add (new Person { name = " 赵 芳 "，sex = " 女 "，age = 22 }); 


D 
CD 
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// 此 时 有 3 个 人 年 龄 大 于 20 岁 


Var temp = from p in people 


where p.age>20 


select p.ToString(); 
Console.WriteLine ("Main 方法 :\t 年 龄 大 于 20 的 人 : "); 


foreach (var s in temp) 


{ 


Console.WriteLine ("Main 方法 :\t"+s); 


i 


Console.WriteLine ("Main 方法 :\t 把 所 有 人 的 年 龄 减 小 5 岁 "); 


// 把 所 有 人 的 年 龄 减 小 5 岁 


people.ForEach(p => p.age -= 5); 


// 此 时 有 1 个 人 年 龄 大 于 20 岁 
foreach (var s in temp) 


Console.WriteLine ("Main 方法 :\t"+s); 


} 


(6) 再 次 运行 程序 ， 此 次 程序 输出 以 下 结果 。 由 输出 结果 可 以 看 出 ， 


输出 都 使 用 


并 不 相同 。 

Main 方法 : 年 龄 大 于 20 的 人 : 
Person.ToString() : 姓名 张 三 

Main 方 法 : ”姓名 : 张 三 性 别 : 男 年 龄 : 23 
Person.ToString() : 姓名 李 四 

Main 方 法: ”姓名 : 李 四 性 别 : 男 年 龄 : 28 
Person.ToString() : 姓名 赵 芳 

Main 方法 : ”姓名 : 赵 芳 性 别 : 女 年 龄 : 22 
Main 方法 : ”把 所 有 人 的 年 龄 减 小 5 岁 
Person.ToString() : 姓名 李 四 

Main 方法 : ”姓名 : 李 四 性 别 : 男 年 龄 : 23 


(7) LINQ 查询 默认 情况 下 是 延迟 执 和 


方法 和 ToArray 方法 ， 如 下 面 代码 所 示 。 


Var temp = (from P in People 


where p.age>20 
select p.ToString( 


)) .ToList() > 


8.3 ”实体 框架 Entity Framework 


虽然 程序 中 两 次 
了 同一 个 查询 ， 但 是 由 于 在 两 次 输出 之 间 对 年 龄 进行 了 修改 ， 得 到 的 输出 结果 


J 的 ， 若 想 让 LINQ 查询 立即 执行 ， 可 以 调用 ToList 


实体 框架 是 ADO.NET 中 的 一 组 支持 开发 面向 数据 的 软件 应 用 程序 技术 。 实体 框架 使 


开发 人 员 可 以 采用 特定 于 域 的 对 象 和 


属性 〈 如 客户 和 客户 地 址 ) 的 形式 使 


数据 ， 而 不 必 


自己 考虑 存储 这 些 数据 的 基础 数据 库 表 和 列 ， 大 大 提升 了 数据 访问 代码 的 通用 性 、 可 读 性 


和 编程 效率 。 
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8.3.1 实体 框架 基本 概念 


数据 建 模 通常 将 数据 模型 分 为 3 个 部 分 : 概念 模型 、 逻 辑 模 型 和 物理 模型 。 概 念 模型 
定义 要 建 模 的 系统 中 的 实体 和 关系 。 关 系数 据 库 的 逻辑 模型 通过 外 键 约束 将 实体 和 关系 规 
范 化 到 表 中 。 物 理 模型 通过 指定 分 区 和 索引 等 存储 详细 信息 实现 特定 数据 引擎 的 功能 。 

物理 模型 由 数据 库 管理 员 进 行 优化 以 改善 性 能 ， 而 编写 应 用 程序 代码 的 程序 员工 作 主 
要 限制 为 通过 编写 SQL 查询 和 调用 存储 过 程 来 处 理 逻 辑 模型 .概念 模型 通常 用 作 捕 获 和 传 
达 应 用 程序 的 要 求 的 工具 。 

实体 框架 可 使 开发 人 员 查 询 概念 模型 中 的 实体 和 关系 ， 同 时 依赖 于 实体 框架 将 这 些 操 
作 转 换 为 特定 于 数据 源 的 命令 。 这 使 应 用 程序 不 再 对 特定 数据 源 具 有 硬 编码 的 依赖 性 。 概 
念 模型 、 存 储 模型 以 及 两 个 模型 之 问 的 映射 以 外 部 规范 《〈 称 为 实体 数据 模型 EDM) 表示 。 
可 以 根据 需要 对 存储 模型 和 映射 进行 更 改 ， 而 不 需要 对 概念 模型 、 数 据 类 或 应 用 程序 代码 
进行 更 改 。 存 储 模 型 是 特定 于 提供 程序 的 ， 因 此 可 以 在 各 种 数据 源 之 问 使 用 一 致 的 概念 
模型 。 

EDM 由 以 下 三 种 模型 和 具有 相应 文件 扩展 名 的 映射 文件 进行 定义 。 

口 概念 架构 定义 语言 文件 〈.csdl) 定义 概念 模型 。 

口 存储 架构 定义 语言 文件 〈.ssdl) 定义 存储 模型 〈 又 称 罗 辑 模型 ) 。 

口 映射 规范 语言 文件 〈.msl) 定义 存储 模型 与 概念 模型 之 间 的 映射 。 


8.3.2 创建 数据 模型 


实体 框架 基于 实体 模型 进行 工作 。 实 体 框 架 实 现 了 模型 到 数据 库 之 间 的 双向 转换 ， 既 
可 以 从 数据 库 生 成 实体 模型 ， 也 可 以 从 实体 模型 自动 生成 数据 库 。 如 果 要 从 数据 库 生 成 实 
体 模型 ， 则 需要 按照 以 下 步骤 进行 操作 。 

(1) 在 Visual Studio 解决 方案 资源 管理 器 中 ， 在 项 目 名 称 上 右 击 ， 从 弹出 菜单 中 选择 
“添加 ”| “新建 项 ”命令 , 在 弹出 的 “添加 新 项 ”对 话 框 中 选择 “ADO.NET 实体 数据 模型 ”， 
如 图 8.4 所 示 。 


图 8.4 添加 ADO.NET 实体 数据 模型 
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(2) 在 图 8.4 中 选择 了 添加 ADONET 实体 数据 模型 以 后 ， 将 会 打开 “实体 数据 模型 
向 导 ” 在 向 导 第 一 步 选择 “从 数据 库 生 成 ” 如 图 8.5 所 示 。 

(3) 在 图 8.5 所 示 向 导 中 单 击 “ 下 一 步 ” 按 钮 ,进入 选择 数据 库 连接 对 话 框 ,选择 SQL 
Server 的 示例 数据 库 Northwind。 

(4) 在 向 导 中 单 击 “ 下 一 步 ”按钮 , 进入 “选择 数据 库 ? 对 象 步骤 。 在 此 选择 Northwind 
表 中 的 Categories、Products、Orders、Order Details 几 个 表 ,， 并 确保 “在 模型 中 加 入 外 键 列 ” 
选项 被 选中 ， 如 图 8.6 所 示 。 


EEEEEETIB Hx 


实体 数 磊 模型 向 导 Ed 


(5) 单 击 


Categary 
区 rier letuils 


图 8.7 生成 的 实体 模型 


从 解决 方案 资源 管理 器 中 可 以 看 到 ， 有 两 个 文件 与 图 8.7 所 示 的 实体 模型 相关 ， 一 个 
是 Modell.edmx， 这 是 一 个 XML 文件 ， 定 义 了 模型 中 包含 的 实体 、 实 体 之 间 的 关系 等 ， 
图 8.7 用 图 形 化 的 方式 显示 了 这 个 文件 的 内 容 。 另 外 一 个 文件 Modell.designercs， 这 是 从 
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实体 模型 生成 的 C# 代 码 ， 包 括 所 有 实体 类 的 代码 、 实 体 之 间 的 关系 代码 等 。 

从 图 8.7 中 可 以 看 出 ， 数 据 库 中 的 表 被 映射 为 实体 类 ， 表 中 字段 被 映射 为 实体 类 的 属 
性 ， 数 据 库 中 的 外 键 关 系 被 映射 为 实体 间 的 导航 属性 。 从 后 台 代 码 中 可 以 看 到 ， 整 个 实体 
模型 派生 自 DataContext 类 ， 而 每 一 个 实体 类 都 派生 自 EntityObject 类 。 


public partial class NorthwindEntities : ObjectContext 
人 

[EdmEntityTypeAttribute (NamespaceName="NorthwindModel", 
Name="Category")] 

[Serializable()] 

[DataContractAttribute (IsReference=true)] 

public partial class Category : EntityObject 

1 


8.3.3 ”查询 数据 


实体 框架 模型 创建 完成 后 , 可 以 使 用 LINQ 语法 对 实体 框架 模型 中 包含 的 表 进行 查询 ， 
查询 得 到 的 结果 通常 为 与 数据 库 对 应 的 实体 类 ， 当 然 也 可 以 在 select 中 使 用 匿名 类 得 到 其 
他 类 型 的 查询 结果 。 

【 例 8-11】 查询 实体 框架 数据 。 

本 例 演示 如 何 使 用 LINQ 对 实体 框架 数据 进行 简单 查询 。 

(1) 创建 一 个 ASPNET Web 应 用 程序 ， 在 项 目 中 添加 前 8.3.2 节 所 创建 的 Northwind 
实体 框架 模型 。 

(2) 在 项 目 中 添加 一 个 SimpleQuery.aspx 页 面 ， 页面 上 放置 一 个 GridView 控件 和 一 个 
分 页 控件 AspNetPager， 页 面 代码 如 下 : 

<form id="forml" runat="server"> 

<div> 

<h3> 简 单 查询 </h3> 


<asp:GridView runat="server" id="grid" AutoGenerateColumns="False" Data- 
KeyNames="ProductID" > 
<Columns> 
<asp:BoundField DataField="ProductID" HeaderText=" 商 品 编号 " 
Readon1y="True" /> 
<asp:BoundField DataField="ProductName" HeaderText=" 商 品名 称 "/> 
<asp:BoundField DataFieldq="SupplierID" HeaderText=" 供 货 商 编号 "” /> 
<asp:BoundField DataField="CategoryID"” HeaderText=" 目 录 编 号 "” /> 
<asp:BoundField DataField="QuantityPerUnit" HeaderText=" 包 装 数 
量 " > 
<asp:BoundField DataField="UnitPrice"” HeaderText=" 包 装 单价 ” /> 
<asp:BoundField DataField="UnitsInStock" HeaderText=" 库 存 数量 " /> 
<asp:BoundField DataField="UnitsOnOrder" HeaderText=" 订 货 数 量 " /> 
</Columns> 
</asp:GridView> 
<webdiyer:AspNetPager ID="pagerl" runat="server" 
onpagechanged="RAspNetPagerl PageChanged"> 
</webdiyer:AspNetPager> 
</div> 
</form> 


(3) 在 SimpleQuery.aspx 页 面 后 台 代码 文件 中 ， 添 加 一 个 bindProducts 方法 以 加 载 并 
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显示 当前 页 面 的 商品 数据 。 


private void bindProducts () 
| 
NorthwindEntities ef=new NorthwindEntities () 7 
var query = from p in ef.Products 
orderby p.ProductID 


select p; // 定 义 查询 
int count = query.Count (); 
pagerl .RecordCount = count; // 设 置 记录 总 数 


var list = query 
.Skip( (pagerl.CurrentPageIndex - 1) * pagerl .PageSize) 
-Take (pagerl .PageSize) 
.ToList (); // 取 得 当前 页 数据 
grid.DataSource = list; 
grid.DataBind(); 
} 


(4) 在 SimpleQuery.aspx 页 面 的 Page_Load 事件 和 分 页 控件 的 PageChanged 事件 中 ， 
调用 上 述 bindProducts() 方 法 以 显示 最 新 页 面 数据 。 


protected void Page Load (object sender, EventArgs e) 


{ 


if (!IsPostBack) 
i 


} 


bindProducts(); 
和 
protected void RspNetPagerl PageChanged (object sender, EventArgs e) 
{ 


) 


bindProducts (); 


(5) 运行 SimpleQuery.aspx 页 面 ， 运 行 结 果 如 图 8.8 所 示 。 


文件 四。 如 枚 @) 查看 MD 历史 外) 书 答 吧 工具 帮助 0) 
[EE Cs we [Ey F 
Googe wmour ME 


简单 查询 
商品 篇 号 商品 名 称 供 货 商 鲍 号 | 目 : 
51 Sir Rodney 2 Seconeo 3 
be (Gustaf a Knickebrod B 
区 Tanchr5d 8 5 
Euarana Fantostica ho 1 
a Tauca Nub-Nougat-Crenelll 日 
日 
3 
Ir 
ls 
日 


ECE 包装 数量 包装 单价 | 简 存 数量 | 


到 pkgs. x 4 pieces |10.0000 |3 


图 8.8 实体 框架 简单 查询 示例 


8.3.4 ”外 键 关 系 和 导航 属性 


在 实体 框架 中 ， 数 据 库 中 的 外 键 关 系 被 映射 为 导航 属性 。 在 Northwind 数据 库 中 ， 产 


=*309” 


第 2 篇 开发 工具 与 第 三 方 框架 


品 表 Product 的 CategoryId 通过 外 键 关联 到 类 别 表 Category 的 CategoryId， 这 个 外 键 关 系 
在 实体 框架 的 模型 图 中 体现 为 Product 模型 和 Category 模型 之 间 有 一 条 连 线 ， 在 Proudct 
这 一 端 标记 为 *， 在 Category 这 一 端 标记 为 0..1， 表 示 商 品 可 以 属于 0 到 1 种 类 别 。 同 时 ， 
在 Product 模型 底部 的 导航 属性 中 有 一 个 Category 导航 属性 ， 通 过 这 个 属性 可 以 找到 产品 
所 属 的 类 别 , 在 Cate 模型 的 导航 属性 中 有 一 个 Products 导航 属性 ， 可 以 找到 该 类 别 的 所 有 
商品 ， 如 图 8.9 所 示 。 


已 属性 
后 categoryID 
Categoryllane | 于 ProdactNane 
村 Deseription | SupplierID 
Pp upp 
村 Picture 于 categoryIm 
日 导航 属性 uantityPerlnit 
园 Produet: 学 WnitPrice 
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图 8.9 实体 框架 中 的 导航 属性 


用 代码 来 描述 图 8.9 所 示 的 导航 属性 的 使 用 如 下 : 
// 下 面 3 行 代码 演示 通过 商品 查找 类 别 


Product p= getAProduc t(); //P 是 一 个 商品 
Category c=p.Category; // 得 到 商品 类 别 
String categoryName=p.Category.CategoryName; // 得 到 商品 所 属 类 别名 称 


// 下 面 两 行 代码 演示 通过 类 别 查 找 商品 

Category c=getACategory(); 

IEnumerable<Product> products=c.Products; 

【 例 8-12】 导航 属性 的 使 用 。 

本 例 演示 如 何 通 过 导航 属性 方便 的 得 到 外 键 关 系 双方 数据 。 

(1) 打开 例 8-11 所 创建 的 项 目 ， 本 例 将 使 用 其 中 创建 的 实体 框架 模型 。 

(2) 在 项 目 中 添加 一 个 页 面 NavigationPropertyPage.aspx, 在 页 面 上 放置 一 个 GridView 
和 一 个 分 页 控件 。 

(3) 在 GridView 中 将 显示 商品 列表 ， 为 GridView 添加 商品 表 的 各 列 ， 并 将 商品 类 别 
设置 为 模板 列 , 该 列 将 显示 类 别名 称 而 非 类 别 编号 。 利 用 实体 框架 的 导航 属性 , 通过 Product 
类 的 Category 属性 可 以 找到 商品 类 别 ， 再 访问 商品 类 别 的 CategoryName 属性 ， 即 可 得 到 
商品 所 属 类 别名 称 。 页 面 代码 如 下 : 


<form id="forml" runat="server"> 
<asp:GridView runat="server" id="grid" AutoGenerateColumns="False" 
DataKeyNames="ProductID" > 
<$--DataGridView 的 各 个 列 ， 共 包括 6 个 数据 绑 定 列 和 一 个 模板 列 
<Columns> 
<asp:BoundField DataField="ProductID" HeaderText=" 商 品 编号 " 
Readonl1y="True" /> 
<asp:BoundField DataField="ProductName" HeaderText=" 商 品名 称 "/> 
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<s-- 模 板 列 ， 显 示 商 品类 别名 称 --$> 
<asp:TemplateField HeaderText=" 商 品类 别 "> 
<ItemTemplate> 
<%#Eval ("Category.CategoryName")$> 
</ItemTemplate> 
</asp:TemplateField> 
<asp:BoundField DataField="QuantityPerUnit"HeaderText=" 包 装 数量 "/> 
<asp:BoundField DataField="UnitPrice" HeaderText=" 包 装 单价 ” /> 
<asp:BoundField DataField="UnitsInStock"” HeaderText=" 库 存 数量 " /> 
<asp:BoundField DataField="UnitsOnOrder"” HeaderText=" 订 货 数 量 " /> 
</Columns> 
</asp:GridView> 
<%-- 分 页 控件 --%> 
<webdiyer:AspNetPager ID="pagerl" runat="server" 
onpagechanged="RAspNetPager1l PageChanged"> 
</webdiyer:AspNetPager> 
</form> 


(4) 在 NavigationPropertyPage.aspx 页 面 的 Page Load 事件 和 分 页 控件 的 PageChanged 
事件 中 编写 代码 加 载 数据 ， 代 码 如 下 : 


public partial class NavigationPropertyPage : System.Web.UI.Page 
{ 


protected void Page Load (object sender, EventArgs e) 

if (!IsPostBack) 

bindProducts(); 

} 

protected void RspNetPagerl PageChanged (object sender, EventArgs e) 
bindProducts (); 


private void bindProducts () // 加 载 数据 


NorthwindEntities ef = new NorthwindEntities () 7 
var query = from p in ef.Products 
orderby p.ProductID 


select p; // 定 义 查询 
int count = query.Count(); 
pagerl.RecordCount = count; // 设 置 记录 总 数 


var list = query 
.Skip( (pagerl.CurrentPageIndex - 1) * pagerl .PageSize) 
.Take (pagerl1 .PageSize) 
ToT // 取 得 当前 页 数据 


grid.DataSource = list; 
grid.DataBind(); 
IEnumerable<Product> t= list[0] .Category.Products; 


3 

(5) 运行 NavigationPropertyPage.aspx 页 面 ， 运 行 结 果 如 图 8.10 所 示 。 

(6) 本 例 前 几 个 步骤 演示 了 从 外 键 表 到 主键 表 的 导航 属性 ， 接 下 再 演示 从 主键 表 到 外 
键 表 的 导航 属性 。 添 加 一 个 新 页 面 NavigationPropertyPage2.aspx， 在 页 面 上 放置 一 个 
DropDownList 控件 和 一 个 GridView 控件 ，DropDownList 用 于 显示 商品 类 别 ，GridView 控 
件 用 于 显示 类 别 下 的 商品 列表 。 页 面 代码 如 下 : 
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<s%--DropDownList 控件 中 显示 所 有 类 别 --%> 
<asp:DropDownList runat="server" id="categoryList" 
DataValueField="CategoryId" DataTextField="CategoryName" Auto-— 
PostBack=True 
onselectedindexchanged="categoryList SelectedIndexChanged"/> 
<br /> 
<asp:GridView runat="server" id="grid" AutoGenerateColumns="False" 
DataKeyNames="ProductID" > 
<Columns> 
<$s--GridView 中 包括 6 个 数据 绑 定 列 --%> 
<asp:BoundField DataField="ProductID" HeaderText=" 商 品 编 号 ”Read- 
Only="True" /> 
<asp:BoundField DataField="ProductName"” HeaderText=" 商 品名 称 "/> 
<asp:BoundField DataField="QuantityPerUnit"HeaderText=" 包 装 数量 "/> 
<asp:BoundField DataField="UnitPrice" HeaderText=" 包 装 单价 ” /> 
<asp:BoundField DataField="UnitsInStock" HeaderText=" 库 存 数量 " /> 
<asp:BoundField DataField="UnitsOnOrder" HeaderText=" 订 货 数 量 " /> 
</Columns> 
</asp:GridView> 
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图 8.10 导航 属性 示例 1 


(7) 在 NavigationPropertyPage2.aspx 页 面 的 PageLoad 事件 中 ， 绑 定 所 有 类 别 到 
DropDownList 控件 。 


protected void Page Load (object sender, EventArgs e) 


if (!IsPostBack) 
{ 
NorthwindEntities ef = new NorthwindEntities(); 
categoryList.DataSource = ef.Categories.ToList();  // 得 到 所 有 类 别 
categoryList.DataBind(); 
} 
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(8) 在 DropDownList 的 SelectedIndexChanged 事件 中 ， 显 示 当 前 选中 类 别 下 的 所 有 商 
品 列表 ， 代 码 如 下 : 

protected void categoryList SelectedIndexChanged (object sender, EventArgs 

e) 


{ 
NorthwindEntities ef = new NorthwindEntities(); 


“3 
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int n = int.Parse (categoryList.SelectedValue) ;// 得 到 所 选择 的 类 别 ID 


Category category = (from c in ef.Categories 

where c.CategoryID == n 

select c 

) -Single(); // 根 据 ID 得 到 类 别 信息 
var list = category.Products; // 得 到 类 别 下 的 所 有 商品 


grid.DataSource = list; 
grid.DataBind(); 
} 


(9) 运行 NavigationPropertyPage2.aspx 页 面 ， 运 行 界 面 如 图 8.11 所 示 。 
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图 8.11 导航 属性 示例 2 


8.3.5 修改 数据 


利用 实体 框架 ， 不 但 可 以 方便 地 检索 数据 ， 而 且 很 容易 实现 添加 、 删 除 和 修改 数据 。 
本 节 将 通过 代码 示例 的 方式 说 明 如 何 添加 、 删 除 和 修改 数据 ， 在 后 面 的 内 容 中 将 介绍 这 种 
机 制 的 原理 。 

【 例 8-13】 添加 、 删 除 、 修 改 数据 。 

本 例 演示 如 何 通 过 实体 框架 实现 添加 、 删 除 、 修 改 商 品 信息 。 

(1) 打开 例 8-11 所 创建 的 项 目 ， 本 例 将 使 用 其 中 创建 的 实体 框架 模型 。 

(2) 在 项 目 中 添加 一 个 页 面 EditProductPage.aspx， 用 以 修改 商品 信息 。 页 面 上 包括 数 
个 TextBox 控件 和 一 个 DropDownList 控件 , DropDownList 用 于 显示 商品 所 属 类 别 , TextBox 
控件 用 于 显示 商品 其 他 属性 。 页 面 项 部 放置 3 个 Button 分 别 实现 更 新 、 删 除 和 添加 功能 。 
EditProductPage.aspx 页 面 代码 如 下 : 

<form id="forml" runat="server"> 

<asp:Button runat="server"” Text=" 更 新 " id="updateButton" /> 

<asp:Button runat="server"” Text=" 删 除 " id="deleteButton" /> 

<asp:Button runat="serVer"” Text=" 添 加 " id="addButton" /> 


<table> 
ET 


<td> 商 品 编号 : </td> 
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<td><asp:TextBox runat="server" id="productId" ReadOn1y="True"/> 
</asp:TextBox></td> 
</tr> 
<tr> 
<td> 商 品名 称 :</td> 
<td><asp:TextBox runat="server" id="productName" /></asp:TextBox> 
</to> 
/Er> 
<tE> 
<td> 所 属 类 别 :</td> 
<td><asp:DropDownList runat="server" id="category" DataValueField= 
"CategoryId" DataTextField="CategoryName" /></asp:DropDownList></td> 
</tr> 
“Er> 
<td> 包 装 数量 :</td> 
<td><asp:TextBox runat="server" id="unitQuantity" /></asp:TextBox> 
</td> 
</tEr> 
<Er> 
<td> 包 装 单价 :</td> 
<td><asp:TextBox runat="server" id="unitPrice" /></asp:TextBox></td> 
</tr> 
<Er> 
<td> 库 存 数量 :</td> 
<td><asp:TextBox runat="server" id="stockQuantity" /></asp:TextBox> 
</td> 
</tr> 
EE 
<td> 已 定货 量 :</td> 
<td> <asp:TextBox runat="server" id="orderUnits"></asp:TextBox> 
</td> 
xr 
<tr> 
<td> 再 定货 点 :</td> 
<td> <asp:TextBox runat="server" id="reorderLevel"></asp:TextBox> 
</td> 
</tr> 
<tr> 
<td> 停 止 供 货 :</td> 
<td><asp:CheckBox runat="server" id="discontinued"></asp:CheckBox> 
</td> 
</Er> 
</table> 
</form> 


(3) 在 Page_ Load 事件 中 ， 根 据 QueryString 所 传递 的 商品 ID 值 ， 从 数据 库 中 检索 对 
应 商品 信息 并 显示 在 页 面 上 。 


protected void Page Load (object sender, EventArgs e) 
{ 
if (!IsPostBack) 

loadData(); 
// 绑 定 商品 类 别 列表 
NorthwindEntities ef = new NorthwindEntities(); 
category.DataSource = ef.Categories; 
category.DataBind(); 
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// 根 据 商 品 ID 从 数据 库 加 载 商品 信息 并 显示 


private void loadData() 


| 


string s = Request.QueryString["id"]; // 从 QueryString 获得 商品 ID 
if (string.IsNullOrEmpty(s)) return; 
int id = int.Parse(s); 
NorthwindEntities ef = new NorthwindEntities(); 
/ /根据 ID 查找 商品 
Product product = (from p in ef.Products 
where p.ProductID == id 
select p) .SingleOrDefault (); 
if (product == null) 
{ 
productName .Text = "[ 未 找到 这 个 产品 ] "; 
return; 


// 将 此 商品 各 个 属性 显示 在 相应 控件 中 

productId.Text = id.ToString(); 

productName.Text = product.ProductName; 
category.SelectedValue = product.CategoryID.ToString(); 
unitQuantity.Text = product.QuantityPerUnit.ToString(); 
unitPrice.Text = product.UnitPrice.ToString(); 
stockQuantity.Text = product.UnitsInstock.ToString(); 
orderUnits.Text = product.UnitsOnOrder.ToString(); 
reorderLevel .Text = product.ReorderLevel.ToString(); 
discontinued.Checked = product.Discontinued; 
ef.Dispose(); 


(4) 在 “更 新 ”按钮 的 Click 事件 中 ， 将 用 户 编辑 后 的 商品 信息 保存 到 数据 库 中 。 注 
意 代码 中 通过 调用 NorthwindEntities 类 的 SaveChanges 方法 实现 数据 保存 功能 。 


protected void updateButton Click(object sender，EventRrgs e) 


NorthwindEntities ef = new NorthwindEntities(); 
int id=Convert.ToInt32 (productId.Text); 


// 根 据 商品 ID 查找 商品 

var product = (from p in ef.Products 
where p.ProductID == id 
select p) 


.SingleOrDefault (); 
// 如 果 商 品 未 找到 则 提示 错误 
if (product == null) 


{ 
Page.ClientScript.RegisterStartupScript (this.GetType(), "showm- 
essage"， "<script>alert (' 数 据 库 中 不 存在 此 id 所 对 应 的 商品 ， 不 能 更 新 。') ; 
</script>"); 
return; 

} 

// 逐 个 设置 商品 的 各 个 属性 


product .ProductName = productName.Text; 

product .CategoryID = int.Parse (category.SelectedValue); 
product .QuantityPerUnit = unitQuantity.Text; 

product .UnitPrice = decimal.Parse (unitPrice.Text); 
product .UnitsInStock = short.Parse(stockQuantity.Text); 
product .UnitsOnOrder = short.Parse(orderUnits.Text); 
product .ReorderLevel = short.Parse (reorderLevel .Text); 
product .Discontinued = discontinued.Checked; 
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h 


// 将 变化 保存 到 数据 库 

ef.SaveChanges (); 

ef.Dispose(); 

Page.ClientScript.RegisterStartupScript (this.GetType(), "showmessa— 


ge, 
"<script>alert (' 商 品 信息 已 经 保存 到 数据 库 ') ;</script>"); 


(5) 在 “删除 ”按钮 的 Click 事件 中 ， 根 据 商品 ID 删除 当前 商品 。 


protected void deleteButton Click(object sender, EventArgs e) 


' 


NorthwindEntities ef = new NorthwindEntities(); 
int id = Convert.ToInt32 (productId.Text); 
// 根 据 商品 ID 查找 商品 
Var product = (from p in ef.Products 
where p.ProductID == id 
select p) 
.SingleOrDefault (); 
// 如 果 商 品 未 找到 则 提示 错误 
if (product == null) 
{ 
Page.ClientScript.RegisterStartupScript (this.GetType(), "showm- 
essage"，"<script>alert(' 数 据 库 中 不 存在 此 ID 所 对 应 的 商品 ， 不 能 更 新 。' ) ; 
</script>")> 
return; 
// 从 实体 框架 上 下 文 删除 商品 并 将 变化 提交 到 数据 库 
ef.DeleteObject (product); 
ef.SaveChanges (); 
ef.Dispose(); 
Page.ClientScript.RegisterStartupScript (this.GetType(), "showmessa-— 
ge"，"<script>alert (' 商 品 已 经 从 数据 库 中 删除 。') ;</script>"); 


(6) 在 “添加 ”按钮 的 Click 事件 中 ， 将 用 户 输入 的 新 商品 信息 保存 到 数据 库 。 


protected void addButton Click(object sender, EventArgs e) 


{ 


:be 


// 创 建 一 个 新 产品 实体 并 根据 用 户 输入 设置 各 个 属性 

Product product = new Product (); 

product .ProductID = int.Parse(productId.Text); 

product .ProductName = productName.Text; 

product .CategoryID = int.Parse (category.SelectedValue); 
product .QuantityPerUnit = unitQuantity.Text; 

product .UnitPrice = decimal.Parse (unitPrice.Text); 
product .UnitsInStock = short.Parse(stockQuantity.Text); 
product .UnitsOnOrder = short.Parse (orderUnits.Text); 
product .ReorderLevel = short.Parse(reorderLevel .Text); 
product .Discontinued = discontinued.Checked; 

// 将 新 产品 添加 到 实体 框架 上 下 文 

NorthwindEntities ef = new NorthwindEntities () 
ef.Products.AddObject (Product) 

// 将 变化 保存 到 数据 库 

ef.SaveChanges (); 

ef.Dispose(); 
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Page.ClientScript.RegisterStartupScript (this.GetType(), "show-— 
message"，"<script>alert (' 新 商品 信息 已 经 添加 到 数据 库 ' ) ;</script>"); 
, 


(7) 运行 EditProductPage.aspx 页 面 ， 运 行 界面 如 图 8.12 所 示 。 
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图 8.12 商品 修改 页 面 运行 界面 


8.4 深入 理解 实体 框架 


8.3 节 中 介绍 了 实体 框架 的 基本 概念 和 基本 操作 ， 通 过 前 面 的 例子 可 以 看 出 ， 使 用 实 
体 框架 只 需 编写 少量 代码 就 可 以 实现 数据 增删 改 查 。 相 对 于 开发 人 员 编 写 的 简单 代码 来 说 ， 
实体 框架 本 身 却 是 一 个 很 复杂 的 框架 ， 封 装 了 对 象 映射 、 状 态 维持 、 连 接管 理 、SQL 语句 
生成 等 众多 功能 。 正 是 由 于 实体 框架 对 这 些 功 能 的 全 面 管 理 和 封装 ， 才 使 得 开发 人 员 编 写 
数据 访问 代码 更 为 简单 。 深 入 理解 实体 框架 的 运行 机 制 对 于 开发 人 员 编写 高 效 的 、 功 能 复 
杂 的 数据 访问 代码 来 说 是 一 个 必要 条 件 。 


8.4.1 对 象 上 下 文 ObjectContext 


对 象 上 下 文 ObjectContext 类 是 以 对 象 这些 对 象 是 EDM 中 定义 的 实体 类 型 的 实例 ) 
的 形式 与 数据 进行 交互 的 主要 类 。ObjectContext 类 的 实例 封装 以 下 内 容 : 

口 到 数据 库 的 连接 ， 以 EntityConnection 对 象 的 形式 封装 。 

口 描述 模型 的 元 数据 ， 以 MetadataWorkspace 对 象 的 形式 封装 。 

口 在 创建 、 更 新 和 删除 操作 过 程 中 跟踪 对 象 的 ObjectStateManager 对 象 。 

在 前 面 的 例子 中 , NorthwindEntities 类 就 是 一 个 对 象 上 下 文 , 继承 自 ObjectContext 类 ， 
可 以 从 自动 生成 的 代码 中 看 到 这 一 点 。ObjectContext 被 创建 时 ， 需 要 指定 其 连接 字符 串 和 
实体 容器 名 称 ， 实 体 容器 名 称 必须 是 唯一 的 ， 可 以 在 实体 模型 设计 视图 的 属性 窗口 中 修改 
这 个 名 称 。 下 面 是 自动 生成 的 NorthwindEntities 类 的 一 部 分 代码 ， 其 中 包括 了 类 继承 层次 
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public partial class NorthwindEntities : ObjectContext 


#region 构造 函数 
// 使 用 应 用 程序 配置 文件 中 名 为 NorthwindEntities 的 连接 字符 串 初始 化 
public NorthwindEntities() : base("name=NorthwindEntities", "North-— 
windEntities") 
ff 
this .ContextOptions.LazyLoadingEnabled = true; // 启 动 延迟 加 载 
OnContextCreated () 


// 初 始 化 新 的 NorthwindEntities 对 象 
public NorthwindEntities (string connectionString) : base (ConnectionS- 
tring, "NorthwindEntities") 
L 
this.ContextOptions.LazyLoadingEnabled = true; 
OnContextCreated(); 
} 
// 初 始 化 新 的 NorthwindEntities 对 象 
public NorthwindEntities (EntityConnection connection) : base (conn- 
ection, "NorthwindEntities") 


this.ContextOptions.LazyLoadingEnabled = true; 
OnContextCreated(); 

| 

#endregion 


一 个 实体 对 象 只 有 在 对 象 上 下 文中 时 ， 才 能 被 更 新 、 删 除 和 修改 。 这 一 点 从 8.3 节 的 
例子 中 也 可 以 看 得 出 。 在 8.3 节 对 商品 数据 进行 修改 的 例子 中 ， 先 对 商品 实体 类 Product 
的 实例 进行 修改 或 者 删除 ， 最 后 调用 对 象 上 下 文 类 NorthwindEntities 的 SaveChanges 方法 
将 修改 保存 到 数据 库 。 如 果 一 个 对 象 不 在 对 象 上 下 文中 ， 则 无 法 将 其 修改 提交 到 数据 库 。 

实体 对 象 既 可 以 在 一 个 对 象 上 下 文中 , 也 可 以 不 在 , 这 两 种 状态 之 间 也 可 以 相互 转换 。 
将 一 个 孤立 的 实体 对 象 添加 到 对 象 上 下 文中 称 为 附加 ， 反 之 ， 使 一 个 对 象 上 下 文中 的 实体 
对 象 脱离 对 象 上 下 文 称 为 分 离 。 在 实体 框架 的 某 个 对 象 上 下 文 内 执行 查询 时 ， 返 回 的 对 象 
会 自动 附加 到 该 对 象 上 下 文 。 如 8.3 节 所 执行 的 查询 操作 都 是 通过 NorthwindEntities 对 象 
上 下 文 进行 的 ， 所 以 查询 到 的 实体 对 象 自动 附加 到 该 对 象 上 下 文 。 而 使 用 new 关键 字 创 建 
的 实体 对 象 并 不 属于 任何 对 象 上 下 文 。 如 下 面 的 例子 所 示 。 


static void Main(string[] args) 


{ 


< 


NorthwindEntities northwind = new NorthwindEntities(); 
//P1 是 从 对 象 上 下 文 northwind 中 查询 得 到 ， 所 以 属于 northwind 
var pl = (from p in northwind.Products 
where p.ProductID==1 
select p) 

a 
// 对 象 p2 使 用 构造 函数 单独 创建 ， 不 属于 任何 对 象 上 下 文 
Product p2 = new Product() { ProductID=1，CategoryID = 1, ProductName 
= "测试 " ]; 
P1.ProductName "新 名 称 1"; 
Pp2.ProductName "新 名 称 2"; 
// 调 用 对 象 上 下 文 的 SaveChanges 方法 时 ， 只 有 pl 的 改变 会 更 新 到 数据 库 
northwind.SaveChanges (); 
northwind.Dispose(); 
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上 


如 果 一 个 实体 对 象 不 属于 任何 对 象 上 下 文 ， 则 可 以 使 用 使 用 下 列 方 法 之 一 将 对 象 附加 
到 对 象 上 下 文 : 


口 


口 


口 
口 


调用 ObjectContext 上 的 AddObject 将 对 象 附加 到 对 象 上 下 文 。 当 对 象 为 数据 源 中 
尚 不 存在 的 新 对 象 时 采用 此 方法 。 

调用 ObjectContext 上 的 Attach 将 对 象 附加 到 对 象 上 下 文 。 当 对 象 已 存在 于 数据 源 
中 但 当前 尚未 附加 到 上 下 文 时 采用 此 方法 。 

调用 ObjectContex 的 AttachTo， 以 将 对 象 附加 到 对 象 上 下 文中 的 特定 实体 集 。 
调用 ObjectContext 上 的 ApplyPropertyChanges 方法 。 当 对 象 已 存在 于 数据 源 中 ， 
并 且 分 离 的 对 象 具 有 希望 保存 的 属性 更 新 时 采用 此 方法 。 如 果 简 单 地 附加 该 对 象 ， 
则 属性 更 改 将 丢失 。 


在 8.3 节 添 加 商品 信息 的 例子 中 ， 就 是 通过 Add 方法 将 新 创建 的 Product 对 象 添加 到 
对 象 上 下 文 ， 如 下 代码 所 示 。 


protected void addButton Click(object sender, EventArgs e) 


{ 


/ /创建 一 个 新 产品 实体 并 根据 用 户 输入 设置 各 个 属性 

Product product = new Product (); 

product .ProductID = int.Parse(productId.Text); 

product .ProductName = productName.Text; 

product .CategoryID = int.Parse (category.SelectedValue) 
Product .QuantityPerUnit = unitQuantity.Text; 

Product .UnitPrice = decimal.Parse (unitPrice.Text) 7 

Product .UnitsInStock = short.Parse (stockQuantity.Text) 
product .UnitsOnOrder short .Parse (orderUnits.Text); 

product .ReorderLevel Short .Parse (reorderLevel .Text); 
product .Discontinued = discontinued.Checked; 

// 将 新 产品 添加 到 实体 框架 上 下 文 

NorthwindEntities ef = new NorthwindEntities () 7 
ef.Products.Addobject (Product) 

// 将 变化 保存 到 数据 库 

ef.SaveChanges (); 

ef.Dispose(); 

Page.ClientScript.RegisterStartupScript (this.GetType(), "showm- 
essage"，"<script>alert(' 新 商品 信息 已 经 添加 到 数据 库 ' ) ;</script>"); 


将 对 象 附 加 到 对 象 上 下 文 时 应 注意 以 下 注意 事项 : 

口 如 果 被 附加 的 对 象 具 有 相关 对 象 ， 则 这 些 对 象 也 被 附加 到 对 象 上 下 文 。 

口 若 要 使 用 Attach 附加 一 个 对 象 , 该 对 象 必须 实现 IEntityWithKey 并 具有 有 效 的 键 。 
口 对 象 以 Unchanged 状态 附加 到 对 象 上 下 文 。 

口 如 果 附 加 的 对 象 不 在 数据 源 中 ， 则 在 执行 SaveChanges 时 不 会 添加 该 对 象 。 在 这 


种 情况 下 ， 如 果 对 属性 进行 了 更 改 ， 则 在 执行 SaveChanges 时 在 服务 器 上 会 引发 
异常 。 若 要 添加 对 象 ， 应 该 使 用 Add() 方 法 而 不 是 Attach0 方 法 。 


如 果 要 将 一 个 当前 存在 于 对 象 上 下 文中 的 对 象 分 离 ， 则 可 以 调用 ObjectContext 类 的 
Detach( 方 法 ， 如 下 代码 所 示 。 


NorthwindEntities northwind = new NorthwindEntities () 


//pl 是 从 对 象 上 下 文 northwind 中 查询 得 到 ， 所 以 属于 northwind 


= 
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var pl = (from p in northwind.Products 
where p.ProductID==1 
select p) 
和 于 全 世人 让 这 
// 将 pl 从 northwind 上 下 文 分 离 


northwind.Detach (pl1); 


8.4.2 对象 状 态 和 对 象 修改 


在 实体 框架 中 ， 实 体 对 象 有 以 下 几 种 状态 : 


口 Detached 状态 : 对 象 存 在 ， 但 没有 
象 上 下 文 之 前 处 于 此 状态 。 通 过 调 


被 对 象 服务 跟踪 。 实 体 在 创建 之 后 且 添加 到 对 
用 Detach 方法 从 上 下 文中 移 除 或 者 如 果 使 用 


NoTrackingMergeOption 加 载 ， 实 体 也 处 于 此 状态 。 


口 Unchanged 状态 : 自 加 载 到 对 象 上 1 
对 象 尚未 修改 。 


下 文中 后 ， 或 自 上 次 调用 SaveChanges 方法 后 ， 


口 Added 状态 : 对 象 添 加 到 对 象 上 1 
AddObject 或 者 Add( 方 法 将 对 象 添 


下 文 ，SaveChanges 方法 尚未 调用 。 通 过 调用 
加 到 对 象 上 下 文 。 


口 Deleted 状态 : 通过 使 用 DeleteObject0 方 法 从 对 象 上 下 文 删除 对 象 后 处 于 此 状态 。 

口 Modified 状态 : 对 象 已 更 改 ， 但 尚未 调用 SaveChanges() 方 法 。 

当 调 用 ObjectContext 类 的 SaveChanges() 方 法 时 ， 实 体 框架 会 根据 对 象 的 状态 生成 对 
应 的 数据 操作 语句 并 执行 。 如 果 对 象 处 于 Added 状态 ， 则 生成 一 条 INSERT 语句 ， 如 果 对 


象 处 于 Deleted 状态 ， 则 生成 一 条 DELETE 


语句 ， 如 果 对 象 处 于 Modified 状态 ， 则 生成 一 


条 UPDATE 语句 。 如 果 对 象 处 于 Unchanged 状态 ， 则 不 需要 处 理 。 如 果 对 象 处 于 Detached 


状态 ， 即 未 附加 到 对 象 上 下 文 ， 则 也 不 会 对 


此 对 象 进行 处 理 。 可 以 通过 SQL Server 提供 的 


监视 工具 SQL Server Profiler 来 查看 Entity Framework 生成 的 SQL 语句 。 


全 提示 : 由 于 Entity Framework 需要 将 C# 的 LINQ 查询 语法 转换 成 对 应 的 SQL 语句 ， 所 


以 有 很 多 C# 方 法 由 于 缺少 对 应 的 


SQL 函数 而 不 能 在 LINQ 中 使 用 。 例 如 ， 以 下 


查询 语句 ffom p in northwind.Products where p.ID=intParse(textBoxl.Textb select p; 


在 执行 时 就 会 抛 出 异常 ， 因 为 Ent 
对 应 的 SQL 函数 。 


ity Framework 无 法 将 intParse 方法 无 法 转换 成 


【 例 8-14】 通过 SQL Server Profiler 查看 Entity Framework 生成 的 代码 。 
(1) 创建 一 个 控制 台 应 用 程序 ， 并 从 Northwind 数据 库 生 成 实体 数据 模型 。 
(2) 在 Main() 方 法 中 编写 代码 ， 对 产品 进行 添加 、 删 除 和 修改 。 


static void Main(string[] args) 


{ 


NorthwindEntities context = new NorthwindEntities(); 


// 从 ObjectContext 查找 一 个 商品 


// 创 建 ObjectContext 


Var query = from p in context.Products 


where p.ProductID 
select p; 


== 1 


Product product = query.First(); 


// 修 改 商 品 并 保存 到 数据 库 


product .ProductName = "新 的 商品 名 称 "; 


30° 
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context -SaveChanges () 

// 创 建 一 个 新 商品 并 保存 到 数据 库 

Product newProduct = new Product(); 
newProduct .ProductName = "新 添加 的 商品 "; 
newProduct -CategoryID = 1; 

newProduct .UnitPrice = 20; 

context .Products.AddObject (newProduct); 
context .SaveChanges (); 

// 删 除 刚才 添加 的 新 商品 

context .Products.DeleteObject (newProduct) 
context .SaveChanges (); 


} 
(3) 启动 SQL Server Profiler 以 监视 SQL 语句 。 从 Windows 程序 菜单 中 选择 Microsoft 


SQL Server 2005 |“ 性 能 工具 ”| SQL Server Profiler， 则 打开 如 图 8.13 所 示 的 SQL Server 
Profiler 主 界面 。 


EET ler 
EDT TT EC 
人 


图 8.13 SQL Server Profiler 主 界面 


(4) 在 SQL Server Profiler 主 界面 中 ， 从 菜单 中 选择 “文件 ”|“ 新 建 跟踪 ”命令 ， 在 
打开 的 连接 对 话 框 中 设置 数据 库 连 接 ， 然 后 进入 如 图 8.14 所 示 的 “跟踪 属性 ”对 话 框 。 在 
其 中 可 以 设置 需要 监视 的 SQL 事件 〈 如 建立 连接 、 执 行 存 储 过 程 、 执 行 SQL 语句 等 )， 本 
例 中 保存 默认 值 即 可 ， 单 击 对 话 框 底部 的 “运行 ”按钮 以 启动 监视 。 


图 8.14 设置 跟踪 属性 
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(5) 启动 SQL Server Profiler 监视 以 后 ， 运 行 刚才 所 编写 的 C# 代 码 ， 并 注意 观察 SQL 
Server Profiler 的 输出 ， 通 过 单 步 执行 C# 代 码 并 同时 对 比 SQL Server Profiler 的 输出 即 可 找 
出 Entity Framework 的 数据 操作 所 生成 的 SQL 代码 。 如 图 8.15 所 示 为 Entity Framework 生 
成 的 添加 商品 的 INSERT 语句 。 
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8.5 小 结 


本 章 介 绍 了 .NET Framework 中 最 新 的 数据 访问 技术 : LINQ to Entity Framework。 首 先 
介绍 了 LINQ 的 基本 操作 ， 然 后 介绍 了 从 数据 库 生 成 实体 对 象 模型 并 检索 和 更 新 数据 ， 最 
后 对 Entity Framework 的 运行 机 制 进行 了 简单 讲解 。 
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在 现在 的 Web 应 用 中 , AJAX 技术 以 其 优秀 的 性 能 和 良好 的 用 户 体验 而 获得 了 人 们 的 
喜爱 ,得 到 了 广泛 的 应 用 .AJAX 的 全 称 是 Asynchronous JavaScript and XML( 异步 JavaScript 
和 XML) ，AJAX 本 身 并 不 是 一 项 新 技术 ， 而 是 多 种 现 有 技术 的 综合 运用 。 随 着 AJAX 技 
术 的 流行 ， 出 现 了 许多 AJAX 框架 ， 目 前 在 ASP.NET 平台 上 开发 AJAX 应 用 主要 有 两 种 
框架 ASP.NET AJAX 和 jQuery。 本 章 和 第 10 章 将 分 别 对 这 两 种 框架 进行 介绍 。 


9.1 AJAX 原理 


虽然 使 用 AJAX 框架 进行 简单 的 开发 不 需要 理解 AJAX 运行 原理 , 但 是 从 提高 开发 人 
员 编程 水 平和 设计 思想 的 角度 来 说 ， 理 解 AJAX 原理 是 很 有 必要 的 。 而 且 ， 如 果 某 些 情况 
下 不 允许 使 用 AJAX 框架 ， 而 是 完全 需要 自己 写 代 码 实现 AJAX， 那 么 AJAX 原理 就 是 必 
须要 掌握 的 一 个 内 容 。 


9.1.1 AJAX 的 意义 


在 传统 的 Web 应 用 程序 中 , 当 用 户 单 击 页 面 上 的 一 个 按钮 时 , 通常 会 引起 页 面 的 提交 ， 
页 面 提交 到 服务 器 端 后 ， 服 务 器 端 对 其 进行 处 理 ， 然 后 再 返回 一 个 页 面 给 浏览 器 ， 浏 览 器 
加 载 并 显示 这 个 页 面 。 从 用 户 的 角度 来 看 ， 单 击 了 页 面 一 个 按钮 ， 页 面 变 成 空白 ， 等 待 若 
干 时 间 ， 页 面 重 新 加 载 并 显示 。 这 种 传统 的 Web 页 面 有 以 下 两 个 特点 : 

(1) 同步 。 从 用 户 提交 数据 到 重新 加 载 新 的 页 面 整 个 过 程 是 同步 进行 的 。 服 务 器 在 处 
理 完 提交 数据 并 返回 新 页 面 以 前 ， 浏 览 器 端 只 能 处 于 等 待 状态 ， 看 不 到 任何 结果 ， 也 不 能 
进行 任何 操作 。 这 种 同步 性 使 得 用 户 等 待 时 间 增 加 。 

(2) 整 页 提交 和 整 页 返回 。 页 面 提交 是 需要 提交 整个 页 面 ， 哪 怕 用 户 只 修改 了 页 面 上 
的 一 个 小 数据 。 服 务 器 返回 页 面 时 也 是 需要 返回 整个 页 面 ， 哪 怕 页 面 上 只 有 一 小 部 分 需要 
更 新 。 这 种 整 页 提交 和 返回 使 得 网 络 传输 量 增 大 ， 而 通常 情况 下 这 种 数据 传输 并 不 是 必 
要 的 。 

AJAX 技术 的 出 现 很 好 地 解决 了 上 述 两 个 问题 。AJAX 采用 异步 方式 与 服务 嚣 端 交 互 ， 
避免 了 用 户 长 时 间 等 待 ， 采用 页 面 局 部 刷新 技术 ， 不 必 提 交 和 返回 整个 页 面 ， 降 低 网 络 流 
量 ， 节 省 网 络 带宽 ， 提 高 下 载 速度 。 
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9.1.2 XMLHttpRequest 对 象 


在 浏览 器 端 使 用 XMLHttpRequest 对 象 与 服务 器 代码 进行 交互 。XMLHttpRequest 对 象 
可 以 向 服务 器 发 送 一 个 HITP 请 求 (GET 或 者 POST)， 并 取得 服务 器 的 响应 结果 。 利 用 
XMLHttpRequest 对 象 的 这 个 功能 ， 在 浏览 器 端 使 用 JavaScript 代码 通过 XMLHttpRequest 
向 服务 器 发 送 请 求 ， 取 得 响应 后 ， 用 JavaScript 显示 在 页 面 上 ， 从 而 实现 了 异步 、 局 部 刷 
新 页 面 的 效果 。 

XMLHttpRequest 对 象 并 没有 完全 统一 的 标准 ,在 各 个 不 同 浏览 器 或 者 同一 浏览 器 的 不 
同 版 本 中 ， 创 建 和 使 用 XMLHttpRequest 对 象 也 有 所 不 同 。 在 正 7 以 前 的 版 本 中 ， 需 要 使 
用 以 下 两 种 语法 之 一 创建 一 个 XMLHttpRequest 对 象 (具体 使 用 哪 种 语法 取决 于 正版 本 )。 

Var request=new ActiveXObject ("Msxm12 .XMLHTTP") ; 

var request=new ActiveXObject ("Microsoft .XMLHTTP") 7 

在 下 7 以 上 版 本 或 者 除 正 以 外 的 多 数 主流 浏览 器 〈 如 FireFox 等 ) 中 ， 都 使 用 
XMLHttpRequest 的 构造 函数 创建 ， 如 下 代码 所 示 。 


var request=new XMLHttpRequest () 


为 了 使 编写 的 代码 兼容 各 种 常见 浏览 器 (包括 正 6)， 应 在 创建 XMLHttpRequest 对 象 
前 检测 浏览 器 是 否 支持 所 调用 的 方法 。 可 以 编写 一 个 辅助 类 及 一 组 方法 来 封装 浏览 器 兼容 
性 检测 及 创建 XMLHttpRequest 对 象 的 功能 ， 如 下 代码 所 示 。 


<script type="text/javascript"> 
//HTTP 这 个 对 象 封装 了 创建 XMLHttpRequest 的 功能 
HTITP = {Ys 
// 定 义 3 个 工厂 函数 ， 分 别 调用 3 种 不 同方 式 创建 XMLHttpRequest 
HTTP. factories = [ 
function () { return new XMLHttpRequest (); }, 
function () { return new ActiveXObject ("Msxml2.XMLHTTP"); }, 
function () { return new ActiveXObject ("Microsoft.XMLHTTP"); } 


日 


HTTP. factory = null; // 适 用 于 当前 浏览 器 的 工厂 函数 
HTTP .newRequest = function () { 
if (HTTP. factory != null) // 如 果 已 经 找到 合适 的 函数 ， 直 接 返 回 


return HTTP. factory(); 
/ /循环 检 测 每 个 工厂 函数 ， 直 到 成 功 创建 一 个 XMLHttpRequest 对 象 
for (var i = 0; i < HTTP. factories.lengthy i++) { 
try { 
var factory = HTTP. factories[i]; 
Var request = factory(); 
if (request != null) { 
HTTP. factory = factory; 
return request; 


} 


catch (e) { continue; } 
} 
// 如 果 未 找到 合适 的 工厂 函数 ， 则 抛 出 异常 
HTTP. factory = function () { 
throw new Error ("当前 浏览 器 不 支持 XMLHttpRequest."); 
让 
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HTTP. factories(); 
ee 
XMLHttpRequest 对 象 有 以 下 主要 属性 。 
onreadystatechange: 请 求 状 态 发 生变 化 时 的 事件 处 理 程序 。 
readyState: 请 求 的 当前 状态 。 
responseBody: 以 字 节 数组 形式 表示 的 服务 器 返回 内 容 。 
responseText: 以 文本 形式 表示 的 服务 器 返回 内 容 。 
responseXML: 服务 器 返回 的 XML 文档 。 
status: 服务 器 返回 的 HITP 状态 代码 。 
statusText: 服务 器 返回 的 HTTP 状态 文本 。 
XMLHttpRequest 对 象 主要 有 以 下 方法 。 
abort: 取消 当前 请 求 。 
口 open: 打开 一 个 请 求 ， 设 置 请 求 的 方法 (GET 或 POST) 、 目 的 URL 等 参数 。 
口 send: 发 送 HITP 请 求 。 
口 setRequestHeader: 设置 HTTP 请 求 头 。 


DOOOODODD 


口 


9.1.3 一 个 简单 的 AJAX 例子 


前 面 介绍 了 AJAX 的 相关 理论 知识 ， 下 面 将 通过 一 个 具体 的 例子 来 说 明 如 何 编写 代码 


使 用 AJAX。 
【 例 9-1】 利用 AJAX 取得 服务 器 当前 时 间 。 


本 例 演示 通过 AJAX 技术 异步 无 刷新 地 取得 服务 器 端 时 间 。 为 了 更 好 地 说 明 AJAX 技 


术 与 普通 Web 技术 的 区 别 ， 本 例 包 括 两 个 页 面 : 一 个 不 采用 AJAX 技术 的 页 面 和 一 个 采 


AJAX 技术 的 页 面 ， 将 二 者 运行 结果 作对 比 ， 能 够 更 直观 地 看 到 AJAX 的 优势 。 


] 


(1) 创建 一 个 HTTP Web 应 用 程序 ， 在 项 目 中 添加 一 个 页 面 TimePage.aspx， 页 面 上 放 
置 一 个 Label 和 一 个 Button，Label 用 于 显示 时 间 ，Button 用 于 从 服务 器 获取 时 间 。 页 面 代 


码 如 下 : 


服务 器 当前 时 间 为 : <asp:Label runat="server" id="labell" Text="Label"></asp: 


Label><br /> 

<asp:Button runat="server" id="ok" Text=" 刷 新 " onclick="ok Click" /> 
//“ 刷 新 ”按钮 的 click 事件 代码 如 下 

protected void ok Click(object sender, EventArgs e) 


// 为 了 使 页 面 延 迟 更 加 明显 ， 线 程 休眠 一 定时 间 

Thread.Sleep(500) > 

labell.Text = DateTime .Now.ToString("T") 7 
} 


(2) 运行 TimePage.aspx， 可 以 看 到 第 一 次 单 击 按钮 ， 页 面 都 需要 等 待 一 段 时 间 。 
(3) 在 项 目 中 添加 一 个 一 般 事 件 处 理 程序 HttpHandler， 用 于 处 理 AJAX 请 求 。 


public class TimeHandler : IHttpHandler 
{ 
public void ProcessRequest (HttpContext context) 
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context .Response.ContentType = "text/plain";  // 内 容 类 型 为 纯 文件 
context .Response.Write (DateTime .Now.ToString("T")); 
// 写 入 当前 时 间 
} 


public bool IsReusable 
上 
get 
{ 
return false; 


} 
h 


(4) 在 项 目 中 添加 一 个 新 HIML 页 面 AjaxTimePage.htm， 页 面 代码 如 下 : 


<head> 
<title>XMLHttpRequest 对 象 示例 </title> 
<script type="text/javascript" src="HttpFactory.js"></script> 
<script type="text/javascript"> 
function test() { 
var request = HTTP.newRequest (); // 创 建 XMLHttpRequest 
request .onreadystatechange = function () { 
// 检 测 ajax 调用 是 否 成 功 
if (request.readyState == 4) { 
if (request.status == 200) 
// 显 示 服 务 器 返回 的 内 容 ( 即 当前 时 间 》 
document .getElementById ("time") .innerHTML = request. 
responseText; 


下 
] 
// 创 建 和 发 送 请 求 


request .open ("GET", "TimeHandler.ashx"); 
request .send (null); 
} 
</script> 
</head> 
<body> 
<div> 
<input type="button" id="bl" onclick="test();" value="TEST" /> 
当前 服务 器 时 间 为 : <span id="time"></span> 
</div> 
</body> 
</html> 


(5) 运行 AjaxTimePage.htm 页 面 ， 单 击 “ 刷 新 ”按钮 以 获得 服务 器 时 间 ， 可 以 看 到 ， 
页 面 能 够 无 刷新 地 获取 并 显示 服务 器 返回 的 时 间 。 


9.2 ASPNET AJAX 基本 控件 


ASPNET AJAX 框架 主要 包括 5 个 核心 控件 ，ScriptManager 控件 、UpdatePanel 控件 、 
UpdateProgress 控件 、Timer 控件 、ScriptManagerProxy 控件 。 通 过 这 些 控件 可 以 方便 地 开 
发 出 基本 的 AJAX 应 用 。 
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9.2.1 ScriptManager 控件 


通过 AJAX 的 全 名 和 9.1 节 的 例子 都 可 以 看 出 , JavaScript 脚本 在 AJAX 中 起 着 至 关 重 
要 的 作用 。 浏 览 器 端 通过 JavaScript 向 服务 器 提交 请 求 、 获 得 响应 并 更 新 页 面 。 在 AJAX 
应 用 中 ， 需 要 编写 大 量 的 JavaScript 代码 ，ASPNET AJAX 的 ScriptManager 控件 是 一 个 
JavaScript 脚本 的 管理 工具 ， 起 着 容纳 、 组 织 、 管 理 JavaScript 脚本 的 作用 。 

在 第 一 个 使 用 ASPNET AJAX 的 页 面 上 都 必须 有 且 只 有 一 个 ScriptManager 控件 ， 而 
且 该 控件 必须 出 现在 其 他 AJAX 控件 之 前 。 在 页 面 上 添加 ScriptManager 控件 的 方法 与 其 他 
控件 相同 ， 从 Visual Studio 工具 箱 的 AJAX Extensions 面板 中 找到 ScriptManager 控件 然后 
拖 动 到 页 面 上 即 可 。 


9.2.2 ScriptManagerProxy 控件 


由 于 每 个 页 面 上 只 能 包含 一 个 ScriptManager 控件 ， 如 果 已 经 在 母 版 页 中 添加 了 该 控 
件 ， 则 在 内 容 页 中 不 能 再 次 添加 。 如 果 需 要 在 内 容 页 中 使 用 ScriptManager 提供 的 功能 ， 这 
时 候 需 要 用 到 ScriptManagerProxy 控件 。 

当 已 在 父 元 素 中 定义 ScriptManager 控件 时 , 使 手套 组 件 ( 如 内 容 页 和 用 户 控 件 ) 可 以 
将 脚本 和 服务 引用 添加 到 页 中 。 该 控件 既 可 以 直接 位 于 页 面 中 ， 也 可 以 间接 位 于 恢 套 组 件 
或 父 组 件 内 部 。 使 用 ScriptManagerProxy 控件 ， 可 在 母 版 页 或 宿主 页 已 包含 ScriptManager 
控件 的 情况 下 ， 将 脚本 和 服务 添加 到 内 容 页 和 用 户 控件 中 。 

ScriptManagerProxy 控件 也 是 在 Visual Studio 工具 箱 的 AJAX Extensions 面板 中 , 使 用 
时 直接 将 其 拖 动 到 页 面 上 即 可 。 


9.2.3 UpdatePanel 控件 


使 用 UpdatePanel 控件 , 无 须 编写 JavaScript 脚本 就 可 以 实现 AJAX 的 异步 局 部 刷新 页 
面 。UpdatePanel 控件 通过 指定 页 中 无 须 刷 新 整个 页 面 即 可 更 新 的 区 域 发 挥 作用 。 此 过 程 由 
ScriptManager 服务 器 控件 和 客户 端 PageRequestManager 类 来 协调 。 当 启用 部 分 页 更 新 时 ， 
控件 可 以 通过 异步 方式 发 布 到 服务 器 。 

通过 使 用 异步 回 发 ， 可 将 页 更 新 限制 为 包含 在 UpdatePanel 控件 中 并 标记 为 要 更 新 的 
页 区 域 。 服 务 器 仅 将 受 影响 元 素 的 HTML 标记 发 送 到 浏览 器 。 在 浏览 器 中 ， 客 户 端 
了 PageRequestManager 类 执行 文档 对 象 模型 (DOM) 操作 以 将 现 有 HTML 替换 为 更 新 的 
标记 。 

UpdatePanel 控件 是 一 个 容器 控件 ， 本 身 没有 任何 可 显示 的 元 素 , 可 以 用 来 在 页 面 上 标 
记 出 可 以 独立 刷新 的 区 域 。UpdatePanel 的 使 用 比较 简单 ， 在 页 面 上 放置 一 个 UpdatePanel 
控件 ， 并 将 需要 局 部 更 新 的 控件 放 到 UpdatePanel 中 就 可 以 了 。 

【 例 9-2】 UpdatePanel 基本 使 用 。 

本 例 演示 UpdatePanel 的 基本 使 用 。 为 了 突出 学 习 UpdatePanel， 减 少 其 他 元 素 十 扰 ， 
此 次 仍 以 获得 服务 器 时 间 为 例 。 
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(1) 创建 一 个 ASPNET Web 应 用 程序 ， 并 添加 一 个 页 面 SimpleUpdatePanel.aspx。 
(2) 在 页 面 上 放置 一 个 ScriptManager 控件 ， 对 应 代码 如 下 : 


<asp:ScriptManager runat="server"> 
</asp:ScriptManager> 


(3) 在 页 面 上 ScriptManager 后 面 添 加 一 个 UpdatePanel 控件 ， 对 应 代码 如 下 : 


<asp:UpdatePanel runat="server"> 
</asp:UpdatePanel> 


(4) 在 页 面 设计 视图 中 ， 从 工具 箱 拖 动 一 个 Label 和 一 个 Button 控件 到 UpdatePanel 
控件 中 ， 如 图 9.1 所 示 ， 页 面 对 应 代码 如 下 : 


<asp:UpdatePanel runat="server"> 
<ContentTemplate> 
当前 时 间 是 : <asp:Label ID="Labell" runat="server" Text=""> 
</asp:Label><br /> 
<asp:Button ID="Button1" runat="server" Text=" 获 得 当前 时 间 " onclick= 
"Buttonl Click" /> 
</ContentTemplate> 
</asp:UpdatePanel> 


于 


口 设计 ] 口 拆 分 | 回 源 | [4 人 Gaiv>][easp: UpastePanel)][ TontentTonplate>] Casp:Label#Labell> [| 


图 9.1 SimpleUpdatePanel 页 面 设计 视图 


(5) 在 “获得 当前 时 间 ” 按 钮 的 Click 事件 中 ， 编 写 以 下 代码 。 

protected void Buttonl Click(object sender, EventArgs e) 

| System.Threading.Thread.Sleep (500); 

Labell.Text = DateTime.Now.ToString("T"); 

(6) 运行 页 面 ， 单 击 “ 获 得 当前 时 间 ” 按 钮 ， 可 以 看 到 ， 页 面 已 经 实现 了 AJAX 效果 。 

例 9-2 是 一 个 最 简单 的 UpdatePanel 使 用 例子 ， 下 面 再 看 一 个 稍微 复杂 一 点 的 例子 。 

【 例 9-3】 使 用 UpdatePanel 无 刷新 更 新 数据 。 

本 例 使 用 UpdatePane 实现 无 刷新 地 加 载 GridView 中 的 数据 。 

(1) 创建 一 个 ASPNET Web 应 用 程序 。 

(2) 在 项 目 中 添加 一 个 从 Northwind 数据 库 生 成 的 实体 框架 数据 模型 ， 其 中 包含 
Products 和 Categories 表 。 

(3) 在 项 目 中 添加 一 个 页 面 UpdatePanelPage.aspx， 在 页 面 上 放置 一 个 ScriptManager 
和 一 个 UpdatePanel 控件 。 

(4) 在 UpdatePanelPage.aspx 页 面 的 UpdatePanel 控件 中 ， 添 加 一 个 GridView 控件 和 
一 个 AspNetPager 分 页 控件 ， 代 码 如 下 : 
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<form id="forml™" 


<div> 
<asp:ScriptManager runat="server"> 
</asp:ScriptManager> 
<asp:UpdatePanel runat="server"> 

<%--UpdatePanel 控件 的 内 容 为 一 个 GridView 和 一 个 AspNetPager--%> 


a 


<ContentTemplate> 


runat="server"> 


<asp:GridView runat="server" ID="grid"></asp:GridView> 


<webdiyer:AspNetPager ID="pager" runat= 


server" onpagechanged= 


"pager PageChanged"></webdiyer:AspNetPager> 
</ContentTemplate> 


/asp:UpdatePanel> 


</div> 
</form> 


(5) 在 页 面 的 Page_Load 事件 中 ， 加 载 第 一 页 数据 。 在 分 页 控件 的 PageChanged 事件 
中 ， 加 载 当 前 面 数据 。 


protected void Page Load (object sender, EventArgs e) 


{ 


下 


1 


f (!IsPostBack) 
loadData (); 


private void loadData() 


{ 
本 


nt size = pager.PageSize ; // 页 面 大 小 
nt index = pager.CurrentPageIndex; // 当 前 页 码 


NorthwindEntities context = new NorthwindEntities();  // 创 建 对 象 上 下 文 
var query = from p in context.Products 


王 


9 
9 


select p; 


nt count = query.Count (); // 得 到 商品 总 数 
pager.RecordCount = count; 

List<Product> list=query.OrderBy (p=>p.ProductID) 

.Skip(size * (index - 1)) 

.Take (size) .ToList (); // 取 得 当前 页 数据 


rid.DataSource = list; 


rid.DataBind(); 


protected void pager PageChanged(object sender, EventArgs e) 


{ 
LL 


} 


oadData (); 


(6) 运行 页 面 ， 单 击 分 页 控件 查看 新 的 一 页 数据 ， 可 以 看 到 能 够 实现 无 刷新 地 加 载 新 


的 数据 。 运 行 界面 如 图 9.2 所 示 。 
在 前 两 个 例子 中 ， 引 起 页 面 


更 新 的 控件 和 被 更 新 的 控件 都 在 同一 个 UpdatePanel 


中 ， 如 例 9-2 中 的 Button 和 Label 在 同一 个 UpdatePanel 中 ， 例 9-3 中 的 GridView 和 
AspNetPager 控件 也 在 同一 个 UpdatePanel 中 。 如 果 引 起 页 面 更 新 的 控件 和 被 更 新 的 控件 不 
在 同一 个 UpdatePanel 中 ， 则 引起 页 面 更 新 的 事件 发 生 时 ， 就 会 变 成 一 个 普通 的 ASPNET 


服务 器 端 寻 


和 件 ， 产 生 整 个 页 面 同步 回 发 。 在 这 种 情况 下， 如果 仍然 想 获 得 AJAX 效果 ， 则 


需要 使 用 UpdatePanel 的 触发 器 。UpdatePanel 的 触发 器 是 指 能 够 触发 UpdatePanel 进行 更 


新 的 事件 。 
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图 9.2 ”UpdatePanel 加 载 数据 


【 例 9-4】 UpdatePanel 触发 器 。 

本 例 演示 UpdatePanel 控件 触发 器 的 使 用 。 
(1) 创建 一 个 ASPNET Web 应 用 程序 。 
(2) 在 项 目 中 添加 一 个 从 Northwind 数据 库 生成 的 实体 框架 数据 模型 ， 其 中 包含 


Products 和 Categories 表 。 


(3) 在 项 目 中 添加 一 个 页 面 TriggerPage.aspx， 在 页 面 上 放置 一 个 ScriptManager 和 一 


个 UpdatePanel 控件 。 


(4) 在 UpdatePanel 控件 中 添加 一 个 GridView 控件 。 
(5) 在 UpdatePanel 外 添加 一 个 AspNetPager 控件 ， 此 时 页 面 代 码 如 下 : 


<div> 


<asp:ScriptManager runat="server"> 

</asp:ScriptManager> 

<asp:UpdatePanel runat="server"> 
<ContentTemplate> 
<asp:GridView runat="server" ID="grid"></asp:GridView> 
</ContentTemplate> 


</asp:UpdatePanel> 


<webdiyer:AspNetPager ID="pager" runat="server" onpagechanged="pager 


PageChanged"> 


</webdiyer:AspNetPager> 


</div> 


(6) 在 页 面 设计 视图 中 ,选中 UpdatePanel 控件 ， 从 


击 右 侧 的 省 略 号 按钮 ， 则 弹出 如 图 9.3 所 示 的 对 话 框 。 
(7) 单 击 “ 添 加 ”按钮 右 侧 下 拉 箭 头 ， 从 下 拉 菜 单 中 选择 AsyncPostBackTrigger ( 异 
步 回 传 触发 器 )， 然 后 从 对 话 框 右 侧 的 ControlID 下 拉 表 中 选择 pager 控件 ， 从 EventName 
中 选择 PageChanged 事件 ， 如 图 9.4 所 示 。 
(8) 设置 完毕 触发 器 后 ， 查 看 页 面 源码 ， 包 含 以 下 代码 。 


<div> 


<asp:ScriptManager runat="server"> 

</asp:ScriptManager> 

<asp:UpdatePanel runat="server"> 
<ContentTemplate> 
<asp:GridView runat="server" ID="grid"></asp:GridView> 
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</ContentTemplate> 
<%- 一 定义 触发 器 --%> 
<Triggers> 


<asp:AsyncPostBackTrigger ControlID="pager" EventName="Page-— 
Changed" /> 
</Triggers> 
</asp:UpdatePanel> 
<webdiyer:AspNetPager ID="pager" runat="server" onpagechanged="pager_ 
PageChanged"></webdiyer:AspNetPager> 
</div> 
</form> 


pdatePanelTrigger 集合 篇 辑 器 


后 站 


成 员 吕 : AsyncPosthack: pager PageChanged ，， 


围 刀 全 
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图 9.3 UpdatePanel Trigger 集合 编辑 器 图 9.4 设置 触发 器 属性 


(9) 运行 TriggerPage.aspx 页 面 ， 运 行 结果 与 例 9-2 相同 。 
9.2.4 UpdateProgress 控件 


UpdateProgress 控件 的 作用 类 似 于 一 个 进度 条 ， 通 常 与 UpdatePanel 控件 配合 使 用 ， 当 
UpdatePanel 正在 更 新 数据 时 ，UpdateProgress 控件 中 的 内 容 (通常 是 更 新 动画 或 者 提示 文 
字 ) 就 会 显示 出 来 , 当 UpdatePanel 更 新 完毕 后 ,UpdateProgress 控件 中 的 内 容 会 自动 隐藏 。 

【 例 9-$】 UpdateProgress 控件 示例 。 

本 例 通过 UpdateProgress 控件 指示 页 面 正 在 进行 局 部 更 新 。 

(1) 打开 例 9-3 或 者 例 9-2 创建 的 项 目 。 

(2) 在 UpdatePanel 前 添加 一 个 UpdateProgress 控件 ， 控 件 中 写 一 段 提 示 文 字 ， 放 一 幅 
加 载 动画 ， 代 码 如 下 : 


<div> 

<asp:ScriptManager runat="server"> 

</asp:ScriptManager> 

<asp:UpdateProgress runat="server" ID="progressl"> 
<$--UpdateProgress 控件 模板 --%> 
<progresstemplate> 
数据 正在 加 载 . . .<br /> 
<img src="images/loading.gif" alt=" 正 在 加 载 ..." style="height: 
100px?™ /> 
</progresstemplate> 

</asp:UpdateProgress> 

<asp:UpdatePanel] runat="server"> 
<ContentTemplate> 
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<asp:GridView runat="server" ID="grid"></asp:GridView> 
<webdiyer:AspNetPager ID="pager" runat="server" onpagechanged= 
"pager PageChanged"></webdiyer:AspNetPager> 
</ContentTemplate> 
</asp:UpdatePanel> 
</div> 


(3) 为 了 使 等 待 加 载 的 动画 效果 更 加 明显 ， 在 loadData() 方 法 中 添加 一 条 语句 ， 使 线 
程 休眠 一 段 时 间 。 


private void loadData() 


. 
System.Threading.Thread.Sleep (1000); 
int size = pager.PageSize ; // 页 面 大 小 
int index = pager.CurrentPageIndex; // 当 前 页 码 


NorthwindEntities context = new NorthwindEntities();  // 创 建 对 象 上 下 文 
Var query = from p in context.Products 
select p; 

int count = query.Count () // 得 到 商品 总 数 
pager.RecordCount = count; 
List<Product> list=query.OrderBy (p=>p.ProductID) 

.Skip(size * (index - 1)) 

-Take (size) .ToList() 7 // 取 得 当前 页 数据 
grid.DataSource = list; 
grid.DataBind(); 


(4) 运行 页 面 ， 运 行 结果 如 图 9.5 所 示 。 
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图 9.5 UpdateProgress 指示 更 新 进度 


9.2.5 Timer 控件 


Timer 控件 可 以 实现 定时 向 服务 器 发 送 请 求 并 触发 服务 器 端 事件 , Timer 控件 通常 用 于 
执行 周期 性 的 任务 。 

【 例 9-6】 自动 更 新 时 间 。 

本 例 演示 通过 Timer 控件 周期 性 自动 获取 并 显示 服务 器 时 间 的 功能 。 本 例 用 到 了 
UpdatePanel 的 Trigger 属性 。 
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(1) 创建 一 个 ASPNET Web 应 用 程序 。 

(2) 在 项 目 中 添加 一 个 页 面 TimerPage.aspx。 

(3) 在 TimerPage 页 面 上 放置 一 个 ScriptManager 控件 、 一 个 UpdatePanel 控件 和 一 个 
Timer 控件 。 在 UpdatePanel 控件 中 放置 一 个 Label 用 以 显示 当前 时 间 。 在 UpdatePanel 外 
面 放置 一 个 Timer 控件 ， 将 Timer 控件 的 Tick 事件 添加 到 UpdatePanel 的 触发 器 中 。 代 码 
如 下 : 


<div> 
<asp:ScriptManager runat="server"> 
</asp:ScriptManager> 
<asp:UpdatePanel runat="server"> 
<%--UpatePanel 内 容 模板 --%> 
<contenttemplate> 
服务 器 当前 时 间 为 : <asp:Label runat="server" id="labell" Text="Label"> 
</asp:Label> 
</contenttemplate> 
<%--UpatePanel 控件 的 触发 器 --%$> 
<triggers> 
<asp:AsyncPostBackTrigger ControlID="timerl" EventName= 
Te /> 
</triggers> 
</asp:UpdatePanel> 
<asp:Timer runat="server" id="timerl" ontick="timerl Tick" Interval= 
"1000"> 
</asp:Timer> 
</div> 


(4) 在 Timer 控件 的 Tick 事件 中 设置 显示 当前 时 间 。 
protected void timerl Tick(object sender, EventArgs e) 


{ 
labell.Text = DateTime.Now.ToString("T"); 


(5) 运行 页 面 ， 可 以 看 到 ， 页 面 每 隔 1 秒 就 会 自动 更 新 服务 器 时 间 。 
9.3 ASP.NET AJAX 控件 工具 箱 简介 


为 了 方便 用 户 开发 功能 更 强大 的 AJAX 应 用 程序 ， 微 软 公司 开发 了 一 个 ASPNET 
AJAX 控件 工具 箱 (ASPNET AJAX Control Toolkit), 其 中 包含 了 一 些 常 用 的 AJAX 功能 和 
控件 。 本 节 将 介绍 ASPNET AJAX 控件 工具 箱 的 用 法 及 几 个 常用 控件 。 


9.3.1 下 载 和 安装 


默认 情况 下 Visual Studio 2010 中 并 不 包含 ASPNET AJAX 控件 工具 箱 , 如 果 想 使 用 的 
话 ， 则 需要 到 网 站 下 载 和 安装 。ASPNET AJAX 控件 工具 箱 官方 网 站 和 下 载 地 址 为 : 
http://ajaxcontroltoolkit.codeplex.com/。 下 载 后 得 到 一 个 压缩 包 ，, 解压 后 有 一 个 dll 文件 和 各 
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个 国家 的 本 地 化 资源 文件 ， 把 这 些 文件 复制 到 磁盘 上 的 一 定 路 径 下 ， 然 后 从 Visual Studio 
中 打开 工具 箱 ， 右 击 工具 箱 空白 处 ， 从 弹出 的 快捷 菜单 中 选择 “选择 项 ”选项 ， 则 会 打开 
如 图 9.6 所 示 的 对 话 框 。 

在 图 9.6 所 示 的 选择 工具 箱 项 对 话 框 中 单 击 “ 浏 览 ” 按 钮 ， 从 弹出 的 新 窗口 中 选择 刚 
才 解 压 以 后 得 到 的 dl 文件 ， 然 后 单 击 “ 确 定 ”按钮 。 此 时 可 以 从 Visual Studio 工具 箱 中 看 
到 新 添加 的 AJAX 工具 了 ， 如 图 9.7 所 示 。 
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图 9.6 “选择 工具 箱 项 ”对 话 框 图 9.7 AJAX 控件 工具 箱 


9.3.2 ”应 用 举例 


ASPNET AJAX Control Toolkit 中 的 组 件 大 致 可 以 分 为 两 类 : 一 类 是 独立 的 控件 , 这 类 
控件 就 像 普 通 控件 一 样 ， 直 接 放 到 页 面 上 使 用 。 另 外 一 类 是 扩展 控件 ， 这 类 控件 本 身 并 不 
是 一 个 独立 的 控件 ， 而 是 通过 对 其 他 现 有 控件 (如 TextBox、Button 等 ) 进行 扩展 来 实现 
功能 。 ASPNET AJAX Control Toolkit 中 包含 很 多 控件 , 限于 篇 幅 , 此 处 不 能 一 一 加 以 介绍 。 
这 里 只 分 别 讲解 两 类 控件 中 的 一 种 ， 其 他 控件 的 使 用 方法 与 此 类 似 ， 具 体 细节 可 参考 微软 
官方 文档 。 

在 使 用 基本 AJAX 控件 时 ， 页 面 上 需要 放置 一 个 ScriptManager 控件 。 与 此 类 似 ， 在 
使 用 AJAX Control Toolkit 控件 里 ， 也 需要 在 页 面 上 放置 一 个 脚本 管理 控件 
ToolkitScriptManager。 

【 例 9-7】 Accordion 控件 。 

Accordion 控件 称 为 可 折 革 面板 控件 ， 可 以 实现 像 QQ 好 友 列 表 那 种 分 组 折 色 的 效果 。 
该 控件 属于 ASPNET AJAX Control Toolkit 中 的 独立 控件 一 类 。 本 例 将 演示 该 控件 的 使 用 
方法 。 

(1) 创建 一 个 ASPNET Web 应 用 程序 ， 在 项 目 中 添加 一 个 页 面 AccordionPage.aspx。 

(2) 在 页 面 上 添加 一 个 ToolkitScriptManagerToolkitScriptManager 控件 , 对 应 代码 如 下 : 
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<asp:ToolkitScriptManager ID="ToolkitScriptManagerl" runat="server"> 
</asp:ToolkitscriptManager> 


(3) 从 工具 箱 中 拖 动 一 个 Accordion 控件 到 页 面 上 ， 对 应 代码 如 下 : 


<asp:Accordion ID="Accordionl" runat="server" > 
</asp:Accordion> 


(4) 为 了 使 Accordion 控件 具有 更 好 的 外 观 效果 ， 需 要 为 其 指定 多 个 CSS 样式 ， 包 括 
面板 标题 样式 HeaderCssClass、 面 板 内 容 样式 ContentCssClass、 选 中 面板 的 标题 样式 
HeaderSelectedCssClass， 代 码 如 下 : 


<asp:Accordion ID="RAccordion1" runat="server" 
HeaderCssClass="accordionHeader" ContentCssClass="accordionContent" 
HeaderSelectedCssClass="accordionHeaderSelected" > 

</asp:Accordion> 


(5) 在 Accordion 控件 中 添加 多 个 面板 。 本 例 添加 了 3 个 面板 ， 分 别 是 基本 信息 、 编 
程 技 术 和 工作 经 验 ， 代 码 如 下 : 


<asp:Accordion ID="Accordionl" runat="server" HeaderCssClass="accordion-— 
Header" ContentCssClass="accordionContent" HeaderSelectedCssClass="acco- 
rdionHeaderSelected" > 
<Panes> 
<asp:AccordionPane ID="AccordionPanel" runat="server"> 
<Header> 基 本 信息 </Header> 
</asp:AccordionPane> 
<asp:AccordionPane ID="AccordionPane2" runat="server"> 
<Header> 编 程 技术 </Header> 
</asp:AccordionPane> 
<asp:AccordionPane ID="AccordionPane3" runat="server"> 
<Header> 工 作 经 验 </Header> 
</asp:AccordionPane> 
</Panes> 
</asp:Accordion> 


(6) 分 别 为 3 个 面板 添加 内 容 ， 代 码 如 下 : 


<asp:Accordion ID="Accordionl" runat="server" HeaderCssClass="accordion- 
Header" ContentCssClass="accordionContent" HeaderSelectedCssClass="acco-— 
rdionHeaderSelected" > 
<Panes> 

<asp:AccordionPane ID="AccordionPanel" runat="server"> 

<Header> 基 本 信息 </Header> 

<Content> 

姓名 : <asp:TextBox ID="TetxtBoxl" runat="server" /><br /> 

年 龄 : <asp:TextBox ID="TextBox2" runat="server" /><br /> 

生日 : <asp:TextBox ID="TextBox3" runat="server" /><br /> 

性 别 : <asp:RadioButton Text=" 男 " ID="male" runat="server" /> 

<asp:RadioButton Text=" 女 " ID="female" runat="server" /><br/> 

学 历 : <asp:TextBox ID="TextBox4" runat="server" /> 

</Content> 

</asp:AccordionPane> 

<asp:AccordionPane ID="AccordionPane2" runat="server"> 
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<Header> 编 程 技术 </Header> 
<Content> 
请 在 这 里 详细 填写 你 所 掌握 的 编程 技术 : <br /> 
<asp:TextBox runat="server" TextMode="MultiLine" Height="200" 
Width="400" /> 
</Content> 
</asp:AccordionPane> 
<asp:AccordionPane ID="AccordionPane3" runat="server"> 
<Header> 工 作 经 验 </Header> 
<Content> 
请 在 这 里 详细 填写 你 的 工作 经 验 和 项 目 经 验 : <br /> 
<asp:TextBox ID="TextBox5" runat="server" TextMode="MultiLine" 
Height="200" Width="400" /> 
</Content> 
</asp:AccordionPane> 
</Panes> 
</asp:Accordion> 


(7) 运行 AccordionPage.aspx 页 面 ， 运 行 结果 如 图 9.8 所 示 。 
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图 9.8 Accordion 控件 示例 


【 例 9-8】 Calender 扩展 控件 。 

ASPNET AJAX Control Toolkit 中 的 日 历 控件 是 一 种 扩展 控件 , 可 以 对 TextBox 进行 扩 
展 使 其 变 成 一 个 日 历 控件 。 本 例 演示 Calendar 扩展 控件 的 使 用 。 

(1) 创建 一 个 ASPNET Web 应 用 程序 ， 添 加 一 个 页 面 AjaxCalendarPage.aspx。 

(2) 在 页 面 上 添加 一 个 ToolkitScriptManagerToolkitScriptManager 控件 。 

(3) 在 AjaxCalendarPage.aspx qt TextBox 控件 。 在 设计 视图 中 ， 选 中 这 
个 TextBox 控件 ， 则 在 控件 右上 角 会 出 现 一 个 箭头 按钮 ， 单 击 这 个 按钮 ， 会 弹出 “TextBox 
任务 ”面板 ， 上 面 有 一 个 “添加 扩展 程序 ” 人 如 图 9.9 所 示 。 
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图 9.9 为 TextBox 添加 扩展 程序 
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全 提示 : 如 果 Visual Studio 2010 开发 环境 中 没有 出 现 图 9.9 所 示 的 “添加 扩展 程序 ”界面 ， 
则 说 明 尚 未 安装 ASPNET AJAX Control Toolkit， 按 照 本 节 开 始 所 讲述 的 方法 
将 ASPNET AJAX Control Toolkit 添加 到 Visual Studio 2010 开发 环境 的 工具 箱 
中 ， 就 会 出 现 这 个 界面 了 。 
(4) 选择 图 9.9 所 示 的 “添加 扩展 程序 ”选项 ， 则 弹出 “扩展 程序 向 导 ” 对 话 框 ， 其 
中 显示 了 可 添加 到 所 选择 的 TextBox 控件 的 所 有 扩展 程序 ， 如 图 9.10 所 示 。 从 中 选择 
CalendarExtender， 并 单 击 “ 确 定 ” 按 钮 。 
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图 9.10 扩展 程序 向 导 


完成 这 一 步 操作 后 ，AjaxCalendarPage.aspx 页 面 代 码 如 下 : 


<asp:ToolkitScriptManager ID="ToolkitscriptManagerl" runat="server"> 

</asp:ToolkitSscriptManager> 

<asp:TextBox ID="TextBoxl" runat="server"></asp:TextBox> 

<asp:CalendarExtender ID="TextBox1l1 CalendarExtender" runat="server" 
Enabled="True" TargetControlID="TextBox1"> 

</asp:CalendarExtender> 


(5) 为 TextBox 添加 了 Calender 扩展 后 ， 运 行 AjaxCalendarPage.aspx 页 面 ， 在 浏览 器 


中 单 击 TextBox 控件 ， 可 以 看 到 ,会 自动 弹出 一 个 日 历 控件 。 在 日 历 控 件 中 选择 一 个 日 期 ， 
所 选中 的 日 期 就 会 自动 设置 到 TextBox 中 。 运 行 界面 如 图 9.11 所 示 。 
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图 9.11 日 期 扩展 控件 示例 
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9.4 小 结 


本 章 介绍 了 AJAX 开发 相关 技术 ， 包 括 AJAX 原理 、ASPNET AJAX 基本 控件 和 
ASPNET AJAX Control Toolkit。 其 中 对 ASPNET AJAX Control Toolkit 只 进行 了 简单 的 介 
绍 , 说 明了 Accordion 控件 和 CalendarExtender 控件 的 简单 应 用 。 更 多 的 控件 及 更 详细 的 使 
用 方法 ， 读 者 可 以 参见 微软 官方 网 站 。 
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随 着 AJAX 技术 的 广泛 应 用 和 有 翌 客户 端 Web 程序 的 逐渐 流行 , Web 开发 人 员 需 要 编写 
越 来 越 多 的 JavaScript 代码 ， 与 服务 器 后 台 进 行 异步 交互 ， 操 作 浏 览 器 页 面 上 的 元 素 。 使 
用 原始 的 JavaScript 代码 完成 这 些 工 作 ， 需 要 编写 大 量 重复 代码 。 为 了 提高 JavaScript 编码 
效率 ， 软 件 开 发 领域 逐渐 出 现 了 一 些 JavaScript 框架 ， 如 jQuery、ExtJs、Prototype 等 。 其 
中 jQuery 是 最 优秀 的 和 应 用 最 广泛 的 JavaScript 框架 之 一 ， 而 且 jQuery 受到 Visual Studio 
IDE 的 支持 , 能 够 显示 智能 提示 。 对 于 ASPNET 开发 人 员 来 说 , 选择 jQuery 作为 JavaScript 
和 AJAX 框架 是 一 个 很 好 的 选择 。 


10.1 jQuery 简介 


jQuery 是 一 个 优秀 的 JavaScript 框架 ， 语 法 简洁 直观 ， 功 能 强大 。 本 节 将 对 jQuery 的 
基础 知识 进行 介绍 ， 为 深入 学 习 打 好 基础 。 


10.1.1 为 什么 使 用 jQuery 


JavaScript 通常 用 来 实现 一 些 动态 网 页 特效 , 例如 当 鼠 标 划 过 某 商 品 时 弹出 一 个 操作 菜 
单 ， 菜 单 上 面 有 “加 入 收藏 "、“ 购 买 ” 等 选项 。JavaScript 还 经 常用 于 和 后 台 服 务 器 交互 并 
显示 得 到 的 结果 ， 例 如 在 省 、 市 、 县 3 个 下 拉 列 表 之 间 建 立 级 联 ， 选 择 一 个 省 ， 则 动态 地 
加 载 该 省 所 有 市 ， 选 择 一 个 市 ， 则 动态 加 载 该 市 的 所 有 县 。 类 似 这 种 工作 ， 如 果 用 原始 的 
JavaScript 来 写 ， 代 码 量 很 大 ， 而 且 重 复 代码 很 多 。 使 用 jQuery 完成 同样 的 功能 可 以 大 大 
减少 代码 量 。 

在 页 面 中 编写 JavaScript 时 ， 比 较 提倡 的 做 法 是 将 JavaScript 脚本 与 HTML 页 面 元 素 
完全 分 离 。 但 是 在 传统 的 JavaScript 编写 中 ， 经 常 需要 将 JavaScript 函数 混在 HIML 标记 
中 。 例 如 ,如 果 要 在 单 击 某 按钮 时 执行 一 个 JavaScript 函数 , 则 通常 是 在 button 标记 的 onclick 
属性 中 指定 函数 名 称 ， 如 下 代码 所 示 。 


<input type="button"” value=" 测 试 按 钮 " onclick="fun1()" /> 


这 种 方式 将 JavaScript 代码 与 HTML 元 素 混 在 一 起 , 不 易 维护 。 而 使 用 jQuery 时 ， 则 
不 需要 在 HTML 元 素 中 通过 onclick 属性 指定 函数 名 称 ， 从 而 将 JavaScript 脚本 与 HTML 
元 素 彻 底 分 离 。 

jQuery 中 还 封装 了 强大 的 AJAX 功能 。 通 过 第 9 章 的 例子 可 以 看 到 ， 如 果 使 用 原始 的 
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JavaScript 脚本 实现 AJAX 功能 ， 编 码 较为 复杂 。 使 用 jQuery 可 以 将 工作 大 大 简化 。 
另外 ， jQuery 是 一 个 开放 的 框架 ,很 容易 对 jQuery 进行 扩展 。 目 前 网 络 上 已 经 有 很 多 
各 种 功能 的 jQuery 插件 ， 对 于 大 多 数 功 能 来 说 ， 开 发 人 员 都 能 够 找到 相应 的 jQuery 插件 。 
总 之 ，jQuery 是 一 个 很 优秀 的 JavaScript 框架 ， 封装 了 很 多 常用 功能 ， 而 且 使 用 简单 ， 
可 读 性 强 ， 其 语法 甚至 比 原始 的 JavaScript 还 要 容易 掌握 。 通 过 jQuery， 开 发 人 员 只 需要 
编写 少量 代码 就 可 以 完成 复杂 的 工作 ， 这 也 正 是 jQuery 宣传 的 口号 : 写 得 更 少 ， 做 得 更 多 
(Write Less, Do More )。 


10.1.2 下载 和 使 用 jQuery 


jQuery 是 作为 一 个 js 文件 来 分 布 的 ， 可 以 从 jQuery 官方 网 站 http://jquery.com/ 下 载 到 
最 新 的 jQuery。jQuery 有 几 个 版 本 ， 作 为 ASPNET 开发 人 员 ， 可 能 用 到 3 个 版 本 ; 即 压缩 
版 、 开 发 版 和 Visual Studio 文档 版 。 其 中 开发 版 是 标准 的 JavaScript 文件 ， 具 有 注释 和 空 
格 、 换 行 。 压 缩 版 是 将 标准 版 中 的 所 有 空白 和 注释 删除 以 后 得 到 的 版 本 ， 这 个 版 本 文件 更 
小 ， 下 载 速度 更 快 ， 能 够 提高 网 页 加 载 速度 。 

Visual Studio 文档 版 是 专门 为 Visual Studio 集成 开发 环境 做 的 版 本 ， 使 用 此 版 本 可 以 
在 Visual Studio 中 显示 jQuery 的 智能 提示 ， 如 图 10.1 所 示 。 


客户 第 对 象 和 事件 -| FW) 
ttl tItley 所 
《script sre="jquery-1.4.2.nin. js” type="text/javascript”>/script> 到 | 


mm a 


加 通 直 | 口 拆 分 [四 还 ] [i[Stn>][ Ser) Cserip 


图 10.1 Visual Studio 对 jQuery 语法 的 智能 提示 


根据 3 个 版 本 的 特点 ， 一 般 在 开发 时 使 用 jQuery 的 压缩 版 和 Visual Studio 文档 版 ， 阅 
读 jQuery 代码 时 可 以 使 用 开发 版 ， 发 布 程序 时 使 用 压缩 版 。 在 页 面 中 引用 jQuery 的 方法 
与 引用 其 他 JavaScript 文件 相同 ， 使 用 script 标记 通过 src 属性 指定 要 加 载 的 文件 ， 如 下 代 
人 码 所 示 。 


<script src="jquery-1.4.2.min.js" type="text/javascript"></script> 


10.1.3 jQuery 和 $ 


jQuery 中 的 $ 是 最 经 常 使 用 的 一 个 函数 。 其 实 jQuery 和 5$ 是 完全 特价 的 ，$ 就 是 jQuery 
的 别名 ， 只 不 过 使 用 $ 语 法 更 简洁 ， 语 句 更 短 。 从 以 下 jQuery 的 源码 中 可 以 明确 地 看 出 $ 
和 jQuery 是 同一 个 函数 。 
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window.jQuery = window.$ = jQuery; 


也 可 以 自己 写 一 段 代 码 来 验证 $ 与 jQuery 相等 ， 代 码 如 下 : 


function funl() { 


if ($ =—= jQuery) 
alert ('$ 与 jQuery 完全 相等 。'); // 运 行 时 将 显示 这 个 提示 
else 


alert ('$ 与 jQuery 不 是 同一 个 对 象 。'); 
} 


根据 参数 的 不 同 ，$ 函 数 主 要 有 5 种 用 法 ， 实 现 5 种 不 同 的 作用 。 
(1) 首先 介绍 $ 函 数 的 第 1 种 用 法 ， 其 作用 为 文档 加 载 完 成 时 执行 一 个 函数 ， 此 时 $ 函 
数 参数 是 一 个 函数 匿名 函数 或 者 命名 函数 )， 其 语法 有 以 下 两 种 : 


$ (function() { /* 在 这 里 写 函数 体 */ }); // 参 数 为 一 个 匿名 函数 
$ (myFunction); // 其 中 myFunction 是 一 个 函数 
下 面 举 例 说 明 $ 函 数 的 这 种 用 法 ， 创 建 一 个 HTML 页 面 并 在 其 中 写 如 下 代码 : 
<html] xmlns="http://www.w3.0org/1999/xhtml"> 
<head> 
<title> jQuery $ 函 数 示 例 </title> 
<script src="jquery-1.4.2.min.js" type="text/javascript"></script> 
<script type="text/javascript"> 
$(function () { 
alert (' 当 HTML 文档 (document) 加 载 完 成 时 ， 就 会 执行 这 段 代码 。' ) ; 
hs 
</script> 
</head> 
<body>jQuery $ 函 数 示例 </body> 


</html> 

运行 上 述 HTML 页 面 ， 则 页 面 加 载 后 立即 弹出 消息 提示 。 

(2)$ 函 数 的 第 2 种 用 法 是 选择 HTML 页 面 上 一 组 元 素 , 并 将 其 作用 jQuery 集合 返回 。 
其 语法 为 : 


$ (expression, context) 


其 中 第 1 个 参数 expression 是 一 个 字符 串 参 数 ， 表 示 查 找 元 素 的 条 件 ， 这 个 参数 通常 
称 为 选择 器 (selector)。 这 个 术语 来 源 于 CSS， 在 CSS 中 ， 如 果 要 设置 某 些 元 素 的 样式 ， 
则 首先 要 找到 这 些 元 素 ， 就 是 通过 选择 器 来 指定 需要 应 用 样式 的 元 素 。jQuery 使 用 与 CSS 
基本 相同 的 选择 器 选择 DOM 元 素 。 第 2 个 参数 指定 了 查找 的 范围 ，$ 函 数 将 在 该 范围 中 查 


找 选 择 器 所 指定 的 元 素 。 如 果 不 指定 第 2 个 参数 ， 则 默认 值 为 document， 即 在 整个 HIML 
文档 范围 内 查找 。 看 一 个 这 种 用 法 的 例子 ， 下 面 的 代码 将 页 面 上 所 有 div 背景 色 设置 为 浅 
绿色 。 


$('div') .css("background-color", "#eeffee"); 

(3) $ 函 数 的 第 3 种 用 法 是 根据 字符 串 生成 相应 的 DOM 元 素 ， 其 语法 为 : 

$ (html) //html 为 一 个 字符 串 ， 表 示 要 创建 的 DOM 元素 
例如 ， 下 面 的 代码 创建 了 一 个 div 元 素 ， 其 中 包含 一 个 p 元 素 。 
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$('<div style="border:dashed 1px silver;"><p> 这 是 一 段 文字 </p> </div>') 
(4) $ 函 数 的 第 4 种 用 法 是 将 一 个 或 多 个 DOM 元 素 封 装 为 jQuery 集合 ， 语 法 如 下 : 
$ (elements) //elements 为 一 个 或 多 个 DOM 元 素 

(5) $ 函 数 的 第 5 种 用 法 是 返回 一 个 空 的 jQuery 集合 ， 这 种 用 法 没有 参数 。 


10.2 ”操作 DOM 元 素 


使 用 JavaScript 或 者 jQuery 最 常见 的 功能 就 是 操作 DOM 元 素 , 包括 读 取 设 置 其 文本 ， 
修改 其 样式 ， 添 加 、 删 除 子 元 素 等 。 本 节 将 介绍 如 何 通过 jQuery 实现 常见 的 DOM 操作。 


10.2.1 处 理事 件 


按照 软件 工程 规范 ， 应 该 把 所 有 JavaScript 代码 与 HTML 元 素 分 离 。 在 使 用 传统 的 
JavaScript 处 理事 件 时 ， 通 过 设置 DOM 元 素 属 性 的 方式 指定 其 事件 处 理 程序 ， 这 样 就 不 可 
避免 地 将 一 部 分 代码 (至 少 是 函数 名 ) 写 进 了 HTML 元 素 中 。 使 用 jQuery 可 以 很 方便 地 
完全 将 代码 与 HTML 元 素 相 分 离 ,将 JavaScript 代码 单独 写 在 <head> 标 记 中 或 者 js 文件 中 。 
使 用 jQuery 为 DOM 元 素 添加 事件 处 理 程序 语法 如 下 : 

$ (选择 器 ) .事件 (函数 ) 


上 述 代码 表示 当选 择 器 所 选中 的 元 素 发 生 指定 事件 时 ， 将 执行 指定 函数 。 例 如 ， 下 面 
的 代码 表示 单 击 buttonl 时 显示 一 个 提示 信息 。 
$('#button1') .click( 


function () 
{ 'hello，jouery-' } 


); 
10.2.2 ”处 理 元 素 内 容 


对 于 DOM 元 素 的 一 个 常见 操作 就 是 读 取 或 者 设置 其 内 容 , 例如， 获取 TextBox 的 值 ， 
设置 div 标记 中 的 内 容 ， 获 取 或 设置 表格 某 单元 格 内 文本 等 。jQuery 主要 提供 两 个 函数 
Chtml0 和 val0〉 来 实现 这 些 功 能 。 

第 1 个 函数 是 html0 函 数 ， 该 函数 有 两 个 作用 ， 如 果 函 数 没 有 参数 ， 则 获取 元 素 内 的 
HTML 内 容 ; 如 果 函 数 有 一 个 字符 串 参数 ， 则 将 元 素 内 容 设 置 为 字符 串 参 数 所 表示 HIML 
元 素 。 其 语法 如 下 : 

html () // 返 回 元 素 内 部 的 HTML 内 容 

html (content) //content 为 字符 串 ， 将 元 素 内 容 设置 为 content 参数 所 指定 值 

【 例 10-1】 使 用 html0 函 数 处 理 元 素 内容 。 

本 例 演示 html0 函 数 读 取 和 设置 元 素 内 容 。 页 面 上 有 两 个 div 和 两 个 按钮 ， 单 击 第 1 
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个 按钮 可 以 实现 读 取 第 1 个 div 内 容 并 以 消息 框 的 形式 显示 出 来 ， 单 击 第 2 个 按钮 可 以 将 
第 1 个 div 的 内 容 复制 到 第 2 个 div 中 。 页 面 运行 结果 如 图 10.2 所 示 。 


lol 


文件 加。 弓 往 加。 查看 书签 加 ”工具 四 ”帮助 加 
Re x /bi0cotnt he 立 - fea a 
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这 是 一 个 起 链接 通 性 Go 
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性 本 


div2 原 来 的 内 容 
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图 10.2 使 用 html0 函 数 处 理 元 素 内 容 


(1) 在 Visual Studio 中 创建 一 个 HIML 页 面 ， 在 页 面 上 放置 两 个 Button 和 两 个 div， 
代码 如 下 : 


<body> 
<div id="main"> 
<input type="button" value=" 显 示 第 1 个 div 内 容 " id="showButton" /><br /> 
<input type="button" value=" 复 制 第 1 个 div 内 容 到 第 2 个 div" id="copyButton" /> 
<div id="div1"> 
<p> 这 是 第 1 个 div 里 的 一 段 文字 </p><br /> 
这 是 一 个 超 链接 通 往 <a href="http://www.google.com">Google (谷歌 ) </a>. 
</div> 
<div id="div2"> 
div2 原来 的 内 容 
</div> 
</div> 
</body> 


(2) 为 了 使 两 个 div 布局 更 加 明显 直观 ， 为 其 添加 CSS 样式 ， 设 置 边 框 和 边 距 。 


<style type="text/css"> 

/* 设 置 div 边框 为 灰色 虚线 ， 设 计 div 间距 ， 使 两 个 div 更 容易 区 分 */ 
div#main > div 

{ border:1lpx dashed silver; margin:10px; height:100px; } 
</style> 


(3) 导入 jQuery 文件 。 

<script src="jquery-1.4.2.min.js" type="text/javascript"></script> 

(4) 编写 jQuery 代码 ， 为 两 个 按钮 的 click 事件 编写 代码 ， 分 别 实现 读 取 和 复制 div 
内 容 的 功能 。 

<script type="text/javascript"> 


$( 


function () { 
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$('#showButton') .click( //showButton 的 click 事件 处 理 程 序 
function () { 

Var content = $("#qdiv1') -html() > // 得 到 divl 的 内 容 

alert (Content) 
Ts 
$('#copyButton') .click( //showButton 的 click 事 件 处 理 程 序 

function () { 
var content = $('#div1') .html(); // 得 到 div1l 的 内 容 


$('#div2') .html (content); // 将 div1 内 容 复制 到 div2 
} 
); 
1 // 最 外 层 function 结束 
ss //$ 结 束 


</script> 


对 于 jQuery 初学 者 来 说 ， 上 述 代码 看 起 来 可 能 有 点 复杂 。$ 函 数 的 参数 是 一 个 函数 ， 
这 个 函数 有 两 个 语句 ， 其 中 每 条 语句 又 是 一 个 函数 (click 语句 )， 这 两 个 click0 函 数 的 参 
数 又 分 别 是 一 个 函数 , 函数 中 包含 多 条 语句 。 像 这 种 函数 多 层 媒 套 的 语法 静态 语言 (如 C#) 
中 并 不 常见 。 为 了 使 初学 者 更 加 清晰 地 理解 上 述 代 码 ， 可 以 将 其 修改 为 以 下 功能 等 价 的 
代码 。 


<script type="text/javascript"> 


S$ DE 二 

function init(){ 
$('#showButton') .click (click1); //showButton 的 click 事件 
$('#copyButton') .click (click2); //copyButton 的 click 事件 

} 

function clickl() { 
var content = $("'#div1').htm]l (); // 得 到 divl 的 内 容 
alert (content); 

} 

function click2() { 
var content = $("'#div1').htm]l (); // 得 到 div1 的 内 容 
$('#div2') .html (content); // 将 divl 内 容 复 制 到 div2 

} 

</script> 


县 提示 : 上 述 两 段 代码 功能 完全 相同 ， 所 不 同 的 是 ， 第 2 段 代码 在 页 面 上 添加 了 3 个 全 局 
函数 ， 而 第 1 段 代码 没有 添加 任何 命名 函数 。 第 2 段 代 码 由 于 添加 了 全 局 命名 函 
数 ， 就 有 可 能 造成 与 其 他 js 文件 中 的 函数 命名 冲突 ， 通常 称 为 污染 命名 空间 。 在 
编写 JavaScript 代码 时 ， 要 尽 可 能 少 地 添加 全 局 成 员 (变量 和 函数 等 ) ， 以 避免 
污染 命名 空间 。 在 本 章 以 后 的 示例 中 ， 将 尽量 使 用 第 1 种 方式 即 匿 名 函数 的 方式 
写 代 码 。 


jQuery 第 2 个 常用 的 操作 元 素 内 容 函 数 为 val0 函 数 。 这 个 函数 也 有 两 种 用 法 ， 如 果 函 
数 没有 参数 ， 则 获取 元 素 的 值 ， 如 果 函 数 有 参数 ， 则 将 元 素 的 值 设置 为 参数 的 值 。val0 函 
数 通常 用 来 获取 和 设置 text、button 等 input 元 素 的 值 .例如 ,假设 用 户 在 一 个 ID 为 yourName 
的 文本 框 中 输入 姓名 ， 则 可 以 使 用 以 下 语句 向 用 户 问 候 。 


Var name = $('#yourName') .val (); 
alert('hello,' + name); 


。344 。 


第 10 章 ”优秀 的 JavaScrip 框架 jQuery 


全 提示 : 并 非 所 有 HTML 元 素 都 支持 val0,， 只 有 在 标记 中 包含 value 属性 中 元 素 ( 如 input 
type=text、input type=button 等 ) 才 能 使 用 val0) 读 取 或 设置 其 值 .对 于 不 支持 value 
属性 的 元 素 (如 div 等) 使 用 val0 函 数 是 没有 意义 的 。 


10.2.3 更改 元素 样式 


jQuery 提供 了 多 种 途径 更 改元 素 的 样式 。 其 中 一 种 最 简单 、 最 直观 的 方式 是 使 用 css() 
函数 。Css0 函 数 可 以 设置 或 获取 元 素 的 css 属性 ， 语 法 如 下 : 

css (name) // 获 取 名 称 为 name 的 css 属性 值 

css (name, value) // 将 名 为 name 的 css 属性 设置 为 value 


下 面 的 代码 是 使 用 css0 函 数 的 例子 。 


Var back=$ ('#div1') .css ("background-color"); // 获 取 divl 的 背景 色 
$('#div2') .css ("background-color", "#eeffee"); /1 设置 div2 背景 色 为 浅 绿色 


使 用 jQuery 设置 元 素 样式 的 第 2 种 方法 为 使 用 addClass0 函 数 和 removeClass( 〇 函数 。 
addClass() 疝 元 素 添加 CSS 类 ，removeClass(0) 函 数 从 元 素 删 除 CSS 类 。 所 添加 的 CSS 类 应 
该 在 页 面 中 定义 。 下 面 的 例子 演示 addClass0 和 removeClass() 函 数 的 使 用 。 

$('#divi') .addClass('green') // 向 divl 添加 green 类 

$('#divl') .removeClass('red'); // 从 div1 删除 green 类 

与 addClass 和 removeClass 密切 相关 的 还 有 一 个 toggleClass 类 ， 其 作用 为 在 添加 和 移 
除 CSS 类 之 间 进 行 切换 。 如 果 元 素 已 经 应 用 了 CSS 类 ， 则 将 其 移 除 ， 和 否则 添加 。 其 用 法 
如 下 : 


$('#div2') .toggleClass('test'); 


现在 网 页 上 经 常见 到 一 种 表格 的 光 棒 效果 ， 即 鼠标 移动 到 表格 某 行 时 ， 该 行 改变 背景 
色 高 亮 显示 ， 鼠 标 移出 时 恢复 成 普通 背景 色 ， 从 而 实现 鼠标 下 的 行 始终 高 亮 显 示 。 可 以 使 
用 toggleClass0) 函 数 结合 mouseenter 和 mouseout 事件 来 实现 这 种 效果 ,但 是 jQuery 还 提供 
了 一 个 更 方便 的 函数 来 实现 这 个 功能 ， 这 个 函数 就 是 hover0 函 数 。hover0 函 数 语法 如 下 : 

hover ( 

function () {/* 这 里 写 鼠 标 进入 时 要 执行 的 代码 */}， 

function () {/* 这 里 写 鼠标 移出 时 要 执行 的 代码 */} 

) 

HoverO 函 数 有 两 个 参数 ， 这 两 个 参数 都 是 函数 ， 当 鼠标 进入 元 素 时 执行 第 1 个 函数 ， 
鼠标 移出 时 执行 第 2 个 函数 。 利 用 hover0 函 数 结合 addClass 和 removeClass0 函 数 , 可 以 方 
便 地 实现 表格 光 棒 效果 。 

【 例 10-2】 表格 效果 示例 。 

本 例 演 示 常 用 的 表格 效果 。 首 先 实现 了 光 棒 效果 ， 随 着 鼠标 移动 ， 鼠 标 下 的 行 总 会 高 
亮 显 示 。 另 外 ， 当 单 击 某 行 时 ， 还 可 以 在 选中 和 非 选中 之 间 进 行 切换 。 

(1) 创建 一 个 ASPNET Web 应 用 程序 。 

(2) 在 项 目 中 添加 Northwind 数据 库 生 成 的 实体 数据 模型 。 
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(3) 在 项 目 中 添加 一 个 页 面 GridEffectaspx， 在 页 面 上 放置 一 个 GridView， 并 在 
Page_Load 中 加 载 数据 。 


protected void Page Load (object sender，EventRrgs e) 
{ 
if (!IsPostBack) 
{ 
// 使 用 LINQ 从 实体 框架 中 查询 数据 
NorthwindEntities northwind = new NorthwindEntities(); 
var query = from p in northwind.Products 
select 
new { p.ProductID, p.ProductName, p.QuantityPerUnit, 
p-UnitPrice, p.UnitsInStock, p.ReorderLevel }; 
// 取 查询 结果 前 10 条 ， 并 绑 定 到 GridView 
var list = query.Take (10) .ToList(); 
grid1.DataSource = list; 
gridl.DataBind(); 


} 
(4) 在 GridEffect.aspx 页 面 中 为 GridView 的 选中 和 高 亮 行 定义 样式 。 


<style type="text/css"> 


.selected { background-color:#ddeeff; } /* 选 中 样式 */ 
.highlight { background-color:#ffffee; } /* 高 亮 样 式 */ 
</style> 


(5) 在 GridEffect.aspx 页 面 中 引入 jQuery。 


<script src="jquery-1.4.2.min.js" type="text/javascript"></script> 
(6) 在 GridEffect.aspx 页 面 中 编写 JavaScript 代码 ， 实 现 选 中 样式 和 光 棒 效果 。 
<script type="text/javascript"> 
$ (function () { 
EO el a // 单 击 时 切换 选中 样式 
function () { 
$ (this) .toggleClass('selected'); 
]) 7 
$('tr') .hover( // 鼠 标 划 过 时 切换 高 亮 样 式 
function () { 
$ (this) .addClass('highlight'); 
}, 
function () { 
$ (this) .removeClass('highlight'); 
} 
); 
1 
) 
</SEE4PE> 


(7) 运行 GridEffect.aspx 页 面 ， 运 行 界 面 如 图 10.3 所 示 。 
10.2.4 隐藏 和 显示 元 素 


在 制作 网 页 时 一 个 常用 的 视觉 效果 就 是 显示 和 隐藏 一 部 分 元 素 。 使 用 jQuery 的 hide0 
和 show0 方 法 可 以 实现 这 个 功能 。 hide0 和 show() 方 法 的 参数 可 以 指定 显示 隐藏 的 速度 ， 从 
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而 达到 合适 的 动画 效果 。hide0 和 show0 函 数 语 法 如 下 : 


hide ( speed, callback ); 
Show ( speed, callback ); 


Gridyiew 效 果 - aezille Firefox 
ELT ET 
9 % [Oh /one 2 立 - Ml 月 
Coogke 可 沼 -+$- M- 昌 -9 -人 -+- 5 © -@- 
Griavior 效 果 = - 
IProductID ProductName QuantityPerUnit |UnitPricelUnitsInStockReorderLevel| 
1 疡 的 商品 名 称 10 boxes x 20 bags |18.0000 
加 (hang 24 =- 12 oz bottles |19.0000 |?8 |5 
I3 |Anisead Syrup 12 - 550 ml bottlesl10. 0000 13 |25 
4 [Chef Anton’ s Cajun Seasonirg (48 -6 oz jars |22.0000 |53 io 
5 [Chef Anton s Gumbo Mix 36 boxes |21.3500 | 0 
6 [Grandna’ s Boysenberry Spread ll2 -8 oz jars |25.0000 |i20 四 
1 lincle Bob’s Organic Dried Fearsl2 - 1 lb pkgs. |30.0000 15 10 
上 horthwoods Cranberry Sauce ll2 -12 oz jars |40.0000 上 io 
日 上 shi Kobe Nilu 18 ~ 500 g pkgs. |97.0000 |29 四 
10 Ikura 12 - 200 ml jars |31.0000 |31 bo 
E23 本 4 


图 10.3 GridView 效果 示例 


两 个 函数 实现 相反 的 功能 ， 具 有 相同 的 参数 。 第 一 个 参数 speed 指定 了 显示 或 隐藏 的 
动画 速度 ， 可 以 指定 slow、normal、fast 之 一 ， 也 可 以 指定 一 个 数值 ， 表 示 完 成 显示 或 隐 
藏 动画 所 需 的 时 间 。 第 二 个 参数 callback 是 一 个 回调 函数 ， 当 显示 或 隐藏 完成 时 会 调用 此 
函数 。 这 是 一 个 可 选 参数 。 

【 例 10-3】 显示 和 隐藏 产品 图 片 。 

(1) 添加 一 个 HTML 页 面 ， 在 页 面 上 放置 一 些 静态 控件 以 描述 图 书信 息 。 

<div> 


书 名 : Design Patterns: Elements of Reusable Object-Oriented Software<br/> 


作者 : Erich Gamma, Richard Helm, Ralph Johnson, John Vissides<br /> 
出 版 社 : 机 械 工业 出 版 社 <br /> 


说 明 : 这 是 四 人 组 (Gung of Four) 写 的 最 经 典 的 设计 模式 著作 ， 英 文 影印 版 <br /> 
<a href="#" id="showImage"> 隐 藏 图 片 </a><br /> 
<div id="bookImage"> 
<img src="images/pl.jpg" alt="" style="width:200px" /> 
</div> 


(2) 在 页 面 中 导入 jQuery。 
(3) 在 页 面 中 编写 代码 ， 当 单 击 “ 隐 藏 /显示 图 片 ”链接 时 ， 以 普通 的 动画 速度 隐藏 或 
者 显示 图 书 封面 图 片 。 


<script type="text/javascript"> 
$ (function () { 
$('#showImage') .click( 
function () { 
// 如 果 当 前 文字 为 “显示 图 片 ”， 则 显示 图 片 ， 并 将 文字 切换 为 “隐藏 图 片 ” 
if ($ (this) -text() == "显示 图 片 ") { 
$ (this) -text ("隐藏 图 片 ") > 
$('#bookImage') .show('normal'); 


上 
// 如 果 当 前 文字 为 “隐藏 图 片 ”， 则 隐藏 图 片 ， 并 将 文字 切换 为 “显示 图 片 ” 
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else { 
$ (this) .text ("显示 图 片 "); 
$('#bookImage') .hide('normal'); 


} 
); 
1D); 
</script> 
(4) 运行 此 页 面 ， 运 行 效果 如 图 10.4 所 示 。 
星 示 和 隐 褒 元 素 - Wozrills Firefo 
文件 加 入 二 看 万 EE 工具 四 帮助 中 
@- CXNH [Ds J/1ocslhost: 1031/HideShowPage. aspx# lr ET p 
Google -+9 M -回避 -， 本:@- 
es 不 可 


Design Patterns: Elements of Reusable Object-Oriented Software 
家 Erich Gamma, Richard Helm, Ralph Johnson, John Vissides 


出 版 社 ， 机 械 工 业 出 版 社 
说 明 ， 这 是 四 人 组 (Gung of Four) 写 的 最 经 典 的 设计 模式 著作 ， 英 文 影印 版 


| http:7/lecalhest:103171G daShowPage aspx# 


图 10.4 显示 和 隐藏 图 片 


10.3 jQuery 常用 选择 器 


jQuery 最 基本 的 功能 是 操作 HIML 页 面 元 素 。jQuery 使 用 类 似 于 CSS 的 选择 器 查找 
页 面 上 的 元 素 ， 然 后 进行 各 种 操作 ， 如 更 改 样式 、 读 取 或 设置 文本 、 隐 藏 显示 等 。jQuery 
支持 多 种 选择 器 ， 常 用 有 以 下 几 种 。 


1. 元 素 选择 器 


元 素 选择 器 能 够 选择 某 一 类 元 素 ， 如 a、img、input 等 。 例 如 ，tr 表示 选择 所 有 tr 元 
素 ，div 表示 选择 所 有 div 元 素 。 如 果 要 给 所 有 表格 加 上 光 棒 效果 ， 即 随 着 鼠标 移动 始终 突 
出 显示 鼠标 下 面 的 行 ， 则 可 以 使 用 以 下 代码 实现 。 

$('tr') .hover( 


function () { $(this).addClass('highlight'); }, 
function () {$ (this) .removeClass('highlight"');} 


); 
上 述 代码 中 SCtr) 就 选择 了 页 面 中 所 有 tr 元 素 , 然后 为 其 鼠标 进入 和 移出 事件 分 别 编写 
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事件 处 理 程序 ， 添 加 和 移 除 高 亮 显示 样式 。 

2. 1D 选 择 器 

ID 选择 器 是 指 根据 一 个 元 素 的 ID 来 选择 元 素 。ID 选择 器 的 语法 是 以 # 开 头 ， 紧 接着 
是 元 素 的 ID。 例 如 $C#button1) 将 选择 页 面 上 ID 为 buttonl 的 元 素 。ID 选择 器 有 时 也 和 元 
素 选 择 器 一 起 使 用 ， 例 如 (div#main") 选 中 ID 为 main 的 div 元 素 。 

3. 类 选择 器 


类 选择 器 可 以 根据 元 素 的 class 属性 进行 选择 。 类 选择 器 的 语法 是 以 .开头 ， 紧 接着 是 
CSS 类 名 。 例 如 ，S$('.important") 将 所 有 页 面 上 所 有 应 用 了 important 类 的 元 素 。 


4. 后 代 选 择 器 


如 果 在 一 个 选择 器 后 面 有 个 空格 ， 再 跟 另 外 一 个 选择 器 ， 则 表示 选择 包含 在 第 1 个 选 
择 器 中 的 第 2 个 选择 器 。 例 如 ,“divp” 选 择 div 中 出 现 的 p 元素。 


5. 子 元 素 选择 器 


如 果 一 个 选择 器 后 面 是 一 个 大 于 号 >， 后 面 再 跟 男 外 一 个 选择 器 ， 则 表示 选择 直接 包 
含 在 第 1 个 选择 器 中 的 第 2 个 选择 器 。 注 意 ， 这 种 选择 器 与 后 代 选 择 嚣 的 不 同 ， 后 代 选 择 
器 只 要 求 后 代 〈 或 者 包含 ) 关系 ， 而 不 管 直接 还 是 间接 ， 而 子 元 素 选择 器 只 选择 直接 子 
元 素 。 


10.4 jQuery+ASP.NET Web Service 实现 AJAX 


在 第 9 半 中 介绍 了 AJAX 基本 原理 ， 使 用 XMLHttpRequest 可 以 实现 AJAX 功能 ， 但 
是 语法 较为 烦琐 。jQuery 对 XMLHttpRequest 进行 了 封装 ， 可 以 更 方便 地 与 服务 器 进行 交 
互 ， 并 取 回 响应 结果 。 再 结合 jQuery 强大 的 操作 DOM 元 素 的 能 力 ， 将 服务 器 返回 结果 及 
时 显示 在 页 面 上 ， 可 以 获得 良好 的 用 户 体验 和 动态 页 面 。 

jQuery 的 ajax0 函 数 封装 了 AJAX 功能 ，ajax0 函 数 语法 如 下 : 

$.ajax( 参数 对 象 ) ; 

其 中 参数 对 象 是 一 个 复杂 的 对 象 ， 包 括 许 多 属性 ， 常 用 的 有 以 下 几 个 属性 。 

口 type: HTITP 请 求 方法 ， 可 以 是 GET 或 者 POST。 

口 url: HITP 语法 的 地 址 , 在 ASPNET 中 , 通常 为 一 个 Web Service 或 者 HttpHandler 

地 址 。 

口 data: 传递 给 服务 器 端的 参数 。 

口 success: ajax 调用 成 功 后 执行 的 回调 函数 。 

【 例 10-4】 验证 用 户 名 。 

在 用 户 注册 时 ， 需 要 验证 新 注册 的 用 户 名 是 否 已 经 存在 ， 如 果 已 经 存在 则 不 允许 再 次 
注册 。 通 过 AJAX 方式 来 完成 这 种 验证 ， 用 户 输入 了 用 户 后 立即 进行 验证 ， 而 不 需要 用 户 
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单 击 按钮 ， 提 高 了 用 户 体验 。 

(1) 创建 一 个 ASPNET Web 应 用 程序 ， 在 项 目 中 添加 一 个 Web Service 用 于 检测 用 户 
名 是 否 可 以 使 用 。Web Service 名 称 为 CheckUserasmx， 其 中 包含 一 个 canUse() 方 法 ， 代 码 
如 下 : 


[WebService (Namespace = "http://tempuri.org/")] 
[WebServiceBinding (ConformsTo = WsiProfiles.BasicProfilel 1)] 
[System.ComponentModel .ToolboxItem (false)] 

// 车 要 允许 从 脚本 中 调用 此 Web 服务 ， 应 添加 下 面 这 行 代码 
[System.Web.Script.Services.ScriptService] 

public class CheckUser : System.Web.Services.WebService 


{ 


// 用 户 名 是 否 可 以 使 用 
[WebMethod] 
public bool canUse (string user) 
string[] existedUsers = { "zhang", "wang", "li", "zhao™" }; 
// 已 经 注册 的 用 户 
for (int i = 0; i < existedUsers.Length; i++) 
{ 
if (user == existedUsers[i]) 
return false; 
} 


return true; 
} 
名 提示: 如 果 需 要 用 JavaScript 调用 ASP.NET Web 服务 ， 不 管 是 用 jQuery， 还 是 直接 用 


XMLHttpRequest， 或 者 使 用 其 他 JavaScript 框架 ，Web 服务 类 前 都 必须 用 
[System.Web.Script.Services.ScriptService] 修 饰 ， 否 则 调用 时 就 会 出 错 。 


(2) 在 项 目 中 添加 一 个 页 面 UserCheckPage.aspx， 页 面 上 放置 用 于 注册 的 各 个 控件 。 
为 简单 起 见 ， 本 例 只 放 了 两 个 TextBox 控件 ， 一 个 用 于 输入 用 户 名 ， 一 个 用 于 输入 密码 。 


<form id="forml" runat="server"> 

<div> 

<h3> 用 户 注册 </h3> 

用 户 名 : <asp:TextBox runat="server" ID="userid"></asp:TextBox> 
<span id="checkUserResult"></span> 

<bre /> 

密码 : <asp:TextBox runat="server" ID="password" TextMode="password"> 
</asp:TextBox><br /> 

省 略 用 户 注册 其 他 信息 . . . 

</div> 

</form> 


(3) 在 UserCheckPage.aspx 页 面 中 引入 jQuery。 


(4) 在 用 户 名 控件 的 blur 事件 (失去 焦点 时 发 生 ) 中 ， 调 用 Web Service 对 输入 的 用 
户 名 进行 检测 ， 并 显示 相应 提示 。 如 果 可 以 使 用 ， 则 用 绿色 字体 提示 ; 如 果 不 可 以 使 用 ， 
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则 用 红色 字体 提示 。 


<script type="text/javascript"> 
$ (function () { 


$ ('#userid') .blur( // 用 户 名 文本 框 失 去 焦点 时 
function () { 
$.ajax({ //AJAX 调用 
type: 'POST', // 使 用 POST 方 法 


url: 'CheckUser.asmx/canUse', 
// 被 调用 的 Web Service 地 址 和 方法 名 
datas: {usere™” + .$9("#userid")-val() Th 
// 传 递 给 Web Service 方法 的 参数 
contentType: "application/json; charset=utf-8", 
dataType: "json", // 使 用 json 传递 数据 
success: function (result) {  // 调 用 成 功 时 执行 此 函数 
if (result.d) { // 若 返回 true， 则 显示 绿色 提示 
$("#checkUserResult') .css('color', "green') 7 
$('"#checkUserResult') .html (' 此 用 户 名 允许 使 用 ' ) ; 
上 
elsent // 若 返回 false， 则 显示 红色 提示 
$('#checkUserResult') .css('color', 'red'); 
$('#checkUserResult') .html (' 此 用 户 名 已 经 被 注册 ， 
不 允许 使 用 。'); 
| 
了 //success function 结束 
error: function () {// 错 误 发 生 时 执行 时 函数 
alert ('Ajax 调用 发 生 错误 ') ; 
} //error function 结束 
} //$.ajax 参数 结束 


//blur function 结束 
} // 最 外 层 function 结束 
); //$ 结 束 
</script> 
(5) 运行 UserCheckPage.aspx 页 面 ， 输 入 用 户 名 ， 再 将 输入 焦点 移动 到 密码 框 ， 则 可 
以 看 到 ， 会 及 时 显示 出 用 户 名 是 否 可 用 的 提示 ， 如 图 10.5 所 示 。 


测试 用 户 名 是 天 可 用 一 Wozilla Firefox EE 
文件 中 总 辑 人 E) 查看 WW) 历史 人) 书签 人 工具 GD) 帮助 如 


Google 起- » +.@- 
DaPsaH 用 [| | 
用 户 注册 

用 户 名 , hang 此 用 户 名 已 经 被 注册 ， 不 允许 使 用 。 
入 


图 10.5 用 户 名 检测 页 面 
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10.5 下 结 


本 章 介绍 了 一 个 当前 很 流行 的 JavaScript 框架 jQuery。jQuery 的 口号 是 “ 写 得 更 少 ， 
做 得 更 多 ”， 能 够 大 大 降低 Web 开发 人 员 的 JavaScript 编码 工作 量 和 难度 。jQuery 使 用 类 
似 于 CSS 的 选择 器 查找 DOM 元 素 , 方便 灵活 。 jQuery 可 以 实现 对 DOM 元 素 的 样式 修改 、 
内 容 访问 、 添 加 、 删 除 等 操作 , 还 可 以 AJAX 方式 与 服务 器 交互 , 功能 强大 。 本 章 仅 对 jQuery 
常见 用 法 进行 了 介绍 ， 如 果 需 要 深入 理解 jQuery， 请 读者 参阅 专门 的 jQuery 书籍 或 者 官方 
网 站 。 
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第 1 章 通用 权限 管理 系统 


在 各 种 类 型 的 管理 信息 系统 中 ， 权 限 管理 是 一 个 普遍 存在 的 问题 。 通 常 一 个 系统 具有 
多 类 用 户 ， 每 类 用 户 具 有 不 同 的 权限 。 例 如 在 网 上 书店 系统 中 ， 未 登录 用 户 可 以 浏览 和 搜 
索 图 书信 息 ， 登 录 的 会 员 用 户 还 可 以 查看 自己 的 订单 情况 ， 系 统管 理 员 可 以 进入 后 台 管理 
页 面 修改 图 书信 息 。 本 章 将 设计 和 实现 一 个 简单 的 ASP.NET 通用 权限 管理 模块 ， 此 模块 
已 经 成 功 应 用 于 作者 开发 的 多 个 商用 软件 项 目 中 。 


11.1 整体 设计 思 


参照 Windows 中 用 户 和 权限 管理 的 方式 ， 结 合 ASP.NET 应 用 程序 的 特点 ， 本 节 所 开 
发 的 通用 权限 管理 基于 角色 进行 权限 分 配 ,基于 页 面 进行 访问 控制 , 基于 HttpModule 进行 
编码 实现 。 本 模块 采用 三 层 结构 进行 设计 和 开发 。 


11.1.1 需求 分 析 


在 一 个 应 用 程序 系统 中 , 根据 权限 的 不 同 , 用 户 可 以 分 为 若干 类 ,例如 有 未 注册 用 户 、 
会 员 用 户 、 系 统管 理 员 等 。 这 种 具有 相同 权限 的 用 户 类 别称 为 一 个 角色 。 系 统管 理 员 可 以 
添加 各 种 角色 ， 并 为 每 种 角色 分 配 不 同 的 权限 。 一 个 用 户 可 以 属于 一 种 或 几 种 角色 ， 从 而 
拥有 所 属 角色 的 所 有 权限 。 

为 了 简化 通用 权限 管理 模块 的 编码 实现 ， 可 以 只 考虑 一 个 用 户 只 属于 一 种 角色 的 情 
况 。 如 果 一 个 用 户 确实 属于 多 种 角色 ， 则 可 以 为 这 个 用 户 分 别 创建 两 个 登录 账号 。 例 如 张 
三 既是 普通 员工 ， 又 是 部 门 经 理 ， 拥 有 不 同 的 权限 ， 那 么 可 以 为 张 三 创 建 两 个 登录 账号 
ZhangSan01 和 ZhangSan02， 分 别 属 于 普通 员工 和 部 门 经 理 的 角色 ， 张 三 可 以 根据 具体 情 
况 选 择 一 个 用 户 账号 进行 登录 。 

在 ASPNET 应 用 程序 中 ， 通 常 一 个 页 面 对 应 程序 中 的 一 个 功能 ， 例 如 ， 图 书 管理 页 
面 对 应 于 图 书信 息 编辑 功能 。 如 果 一 个 用 户 或 者 角色 拥有 使 用 某 个 功能 的 权限 ， 那 么 这 个 
用 户 或 者 角色 允许 访问 相应 页 面 ， 否 则 不 允许 访问 相应 页 面 。 根 据 这 种 思路 ， 可 以 通过 允 
许 和 禁止 用 户 访 问 页 面 来 实现 权限 控制 。 当 用 户 要 访问 某 一 个 页 面 时 ， 首 先 判 断 此 用 户 是 
否 拥 有 访问 此 页 面 的 权限 ， 如 果 有 则 继续 访问 ， 如 果 没 有 则 提示 权限 不 足 。 

在 实现 应 用 中 ， 还 经 常 遇 到 这 样 一 种 情况 ， 即 有 些 页 面 允许 所 有 用 户 访问 ， 例 如 网 上 
书店 中 的 图 书 浏 览 和 搜索 页 面 。 对 于 这 种 页 面 ， 不 需要 判断 用 户 权 限 ， 可 以 直接 访问 。 
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11.1.2 ”数据 库 结构 设计 


根据 前 面 的 功能 分 析 和 设计 思路 ， 通 用 权限 管理 模块 数据 库 中 共 包含 4 个 表 : 用 户 表 
User、 角 色 表 User Role、 功 能 模块 表 ApplicationModule 和 角色 权限 表 RoleRight， 表 中 字 
段 类 型 及 含义 如 下 。 


1. 


日 BB BDODGD O96G VM goa 


角色 表 User 


ID: 角色 ID，nvarchar(10) 类 型 ， 主 键 。 
Name: 角色 名 称 ，nvarchar(10) 类 型 ， 不 可 空 。 
Description: 角色 描述 ，nvarchar(50) 类 型 ， 可 空 。 


用 户 表 User 


ID: 用 户 ID， 登 录 时 使 用 ，nvarchar(10) 类 型 ， 主 键 。 

Password: 登录 密码 ，nvarchar(20) 类 型 ， 不 可 为 空 ， 默 认 值 “123456”。 
UserNamer: 用 户 名 称 ，nvarchar(10) 类 型 ， 不 可 为 空 。 

RoleID: 用 户 所 属 角色 ID， 外 键 关 联 到 角色 表 UserRole。 


功能 模块 表 ApplicationModule 


ID: 模块 ID，nvarchar(10) 类 型 ， 主 键 。 

Name: 模块 名 称 ，nvarchar(30) 类 型 ， 不 可 为 空 。 

URL: 模块 所 对 应 的 页 面 地 址 ，nvarchar(50) 类 型 ， 不 可 为 空 。 

Description: 对 于 此 功能 模块 的 描述 ，nvarchar(80) 类 型 。 

IsPublic: 是 否 公共 模块 ，bit 类 型 。 公 共 模 块 表示 所 有 用 户 都 可 以 访问 的 模块 。 


角色 权限 表 RoleRight 


RoleID: 角色 ID， 外 键 关 联 到 UserRole 表 。 
ModuleID: 模块 ID， 外 键 关联 到 ApplicationModule 表 。 
TheRight: 角色 对 模块 的 权限 ，nchar(1) 类 型 ，0 表示 无 权 ，1 表示 有 权 。 


以 上 各 表 之 间 关系 如 图 11.1 所 示 。 
icationModule 
| ET | 证 和 | a | + 认 字 
下 mvarshar(10) 了 vanchar(l0) 
Passward Marchar(2D) [a] | um mvarcharGa0) Cc 
LserNane mvarchar(10) [ol | rr varchar(50) [e] 
Pei marcha(10) D | ==eao varchar(e0) 区 
加 面世 一 区 5 
| 
UserRole RoleRight 
3 数 拓 关 型 “| if | E 
了 mwarcnartlO) 口 | RoleID rvarchar(10) 口 
Name nvarchar(10) [el ModuleID nvarchar(10) [s] 
Deccription Mvarchar(S0) a 夯 | They chal) F 
站 


图 11.1 数据 库 结构 
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11.1.3 ”搭建 项 目 框架 


本 节 所 设计 开发 的 权限 管理 模块 是 一 个 通用 模块 ， 可 以 应 用 于 不 同 需求 的 ASPNET 
应 用 程序 中 。 为 了 提高 程序 通用 性 ， 在 Visual Studio 中 使 用 一 个 单独 的 解决 方案 实现 通用 
权限 管理 模块 。 该 模块 采用 三 层 结构 设计 ， 使 用 实体 框架 完成 数据 访问 ， 解 决 方案 中 应 该 
包含 4 个 项 目 : 实体 框架 层 、 数 据 访问 层 、 业 务 逻 辑 层 和 Web 表现 层 。 搭 建 整 个 解决 方案 
框架 的 步骤 如 下 : 

(1) 创建 一 个 空白 解决 方案 。 

(2) 在 解决 方案 中 添加 一 个 类 库 项 目 STL.Entity 作为 实体 框架 层 。 

(3) 在 解决 方案 中 添加 一 个 类 库 项 目 SI 世 .Dal 作为 数据 访问 层 。 

(4) 在 解决 方案 中 添加 一 个 类 库 项 目 SL.BIl 作为 业务 轴 辑 层 。 

(5) 在 解决 方案 中 添加 一 个 ASPNET Web 应 用 程序 项 目 SIL.Web 作为 表现 层 。 

(6) 在 解决 方案 中 添加 一 个 类 库 项 目 SIL.Common， 此 项 目 中 包含 会 在 三 层 用 到 的 公 
共 类 。 

(7) 在 各 个 项 目 之 间 添 加 引用 关系 ， 其 他 3 个 项 目 都 引用 实体 框架 层 和 公共 类 库 ， 业 
务 逻 辑 层 引 用 数据 访问 层 ， 表 现 层 引用 业务 逻辑 层 。 


11.2 公共 类 库 和 实体 框架 


根据 11.1 节 的 分 析 ， 通用 权限 管理 系统 采用 三 层 结构 ， 使 用 实体 框架 对 业务 实体 类 进 
行 封 装 ， 并 作为 数据 访问 的 基础 。 另 外 ， 项 目 中 还 有 一 个 公共 类 库 ， 其 中 包含 一 些 公共 代 
码 。 本 节 将 介绍 实体 框架 和 公共 类 库 的 实现 。 


11.2.1 公共 类 库 的 实现 


公共 类 库 SJL.Common 中 包含 了 在 数据 访问 层 、 业 务 轴 辑 层 、 表 现 层 都 会 用 到 的 类 ， 
这 些 类 还 会 应 用 于 除 通用 权限 管理 以 外 的 其 他 项 目 中 ， 所 以 这 些 类 不 能 包含 在 通用 权限 管 
理 项 目的 实体 框架 层 中 ， 而 是 要 单独 存储 于 一 个 类 库 ， 以 便于 在 其 他 项 目 中 复 用 。 

公共 类 库 中 主要 包含 两 个 类 : 一 个 分 页 数据 检索 参数 类 PageDataArgument， 用 于 指定 
分 页 检索 数据 时 的 分 页 参数 ， 如 页 号 、 页 面 大 小 等 ， 一 个 日 期 范围 类 DateRange， 用 于 指 
定 按照 日 期 检索 时 的 日 期 范围 。PageDataArgument 类 的 代码 如 下 : 


// 进 行 分 页 数据 查询 时 的 分 页 参数 
public class PageDataArgument 
. 


#region 字段 

public int pageIndex=0; // 页 码 

public int pageSize = 10; // 页 面 大 小 
public bool refreshCount = false; / /是否 更 新 记录 总 数 
public int count = -1; // 记 录 总 数 


private static PageDataArgument defaultArg = new PageDataArgument (); 


“be 
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} 


// 默 认 分 页 参数 

private static PageDataArgument allDataArg 

= new PageDataArgument() { pageSize = 99999 }; 

// 得 到 所 有 数据 的 分 页 参数 

#endregion 
#region 构造 函数 
public PageDataArgument() { } 
public PageDataArgument (int index, int size, bool getCount) 
{ 

pageIndex = index; 

pageSize = size; 

refreshCount = getCount; 
} 
#endregion 
#region 属性 
public static PageDataArgument allData 
{ get { return allDataArg; } } 
public static PageDataArgument defaultValue 
{ get { return defaultArg; }} 
#endregion 


日 期 范围 类 DateRange 代码 如 下 : 
// 日 期 范围 类 ， 用 于 指定 查询 时 的 起 止 日 期 


public class DateRange 


{ 


// 以 下 两 个 字段 定义 了 起 始 日 期 ( 含 ) 和 结束 日 期 ( 含 ) 
public DateTime from; 

public DateTime to; 

///<summary> 

/7/ 根 据 两 个 日 期 (忽略 时 间 〉 生 成 日 期 范围 
///</summary> 

///<param name="day1"> 起 始 日 期 </param> 
///<param name="day2"> 终 止 日 期 </param> 
///<returns></returns> 


///<remarks> 用 户 在 查询 时 习惯 于 输入 两 个 日 期 (不 含 时 间 〉， 而 查询 时 需要 查询 包括 起 


/// 止 日 期 在 内 的 时 间 范 围 。 如 输入 2010-1-1 和 2010-1-31， 

/// 则 得 到 的 日 期 范围 是 2010-1-1 00:00:00 至 2010-1-31 23:59:59 
/// 这 个 方法 能 够 方便 地 根据 日 期 生成 用 户 需 要 的 时 间 范 围 
///</remarks> 


public static DateRange between2Date (DateTime dayl, DateTime day2) 


DateRange a = new DateRange () 7 
a.from = dayl.Date; 
a.to = day2.Date.AddDays (1) .AddSeconds (-1); 
return a; 
tL 
// 功 能 同上 ， 根 据 两 日 期 生成 时 间 范 围 ， 但 是 两 个 日 期 参数 是 string 类 型 
public static DateRange between2Date (string dayl, string day2) 
{ 
return between2Date (DateTime.Parse (dayl), DateTime.Parse (day 


| 
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11.2.2 ”实体 框架 层 


本 模块 使 用 实体 框架 实现 数据 访问 ， 构 建 实体 框 架 层 的 步骤 如 下 : 

(1) 打开 实体 框架 层 项 目 SIL.Entity， 从 菜单 中 选择 “项 目 ”| “添加 新 项 ”命令 ， 从 
弹出 的 “添加 新 项 ”对 话 框 中 选择 “ADO.NET 实体 数据 模型 ”命名 为 UserRight。 单 击 “ 确 
定 ” 按 钮 ， 则 弹出 如 图 11.2 所 示 的 “实体 数据 模型 向 导 ” 对 话 框 ， 在 其 中 选择 “从 数据 库 
生成 ”模型 内 容 。 

(2) 在 “实体 数据 模型 向 导 ” 对 话 框 中 单 击 “ 下 一 步 ” 按 钮 ， 配 置 适 当 的 数据 库 连 接 ， 
再 单 击 “ 下 一 步 ”按钮 ， 出 现 如 图 11.3 所 示 的 “选择 数据 库 对 象 ”对 话 框 。 在 此 选中 数据 
库 中 4 个 表 ， 并 选中 “在 模型 中 包含 外 键 列 ” 选 项 ， 然 后 单 击 “ 完 成 ”按钮 以 完成 实体 数 
据 模型 向 导 。 

(3) “实体 数据 模型 ”向 导 完成 后 ， 在 项 目 中 生成 如 图 11.4 所 示 的 实体 框架 类 。 


EEEEEEEB x| 


实体 数 磊 模型 向 导 


we 


名 要 在 模 列 中 包 馈 各 些 茹 可 序 对 铺 ()? 


模型 命名 空间 人 
jlFr ereriodel] 


国王 了 EE 取消 《< 上- 步 四 | 下 -四 取消 
图 11.2 实体 数据 模型 内 容 图 11.3 选择 数据 库 对 象 
守 Veer A 2 Vserkole 习 RoleRiaht 习 oe Applicationlodole A 


Hl Properties = Properties = Properties 3 Properties 
河 了 呵 了 晤 rolem 虹 m 
部 password 本 Nane noduleID 可 Nane 
守 UeerNane 1 本 Description 1 *| 喀 IheRight 六 1 村 号 
分 RoleD = Navigation Prop… Navigation Prop… 加 Description 
EB Navigation Pr… 国 User 加 Applicationl*… 分 IsPublic 
纪 UserRole 辐 RoleRight 图 UserRole 3 Navigation Properties 
恒 RoleRight 


图 11.4 实体 框架 类 


11.3 数据 管理 


通用 权限 管理 模块 涉及 到 用 户 、 角 色 、 模 块 各 方面 的 数据 。 用 户 可 以 对 这 些 数 据 进 行 
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浏览 修改、 添加 、 删 除 等 操作 ， 本 节 将 介绍 这 些 功 能 的 实现 。 
11.3.1 角色 管理 


角色 管理 功能 可 以 实现 添加 、 删 除 和 修改 用 户 角色 。 在 三 层 结 构 的 程序 中 ， 角 色 管 理 
功能 需要 三 层 配合 实现 ， 具 体 步 又 如 下 。 

(1) 在 数据 访问 层 SJL.Dal 中 添加 一 个 类 UserRoleDal, 在 其 中 实现 用 户 角色 的 增 、 删 、 
改 、 查 的 功能 ， 代 码 如 下 : 


// 用 户 角色 数据 访问 层 
public static class UserRoleDal 


{ 


/// <summary> 

/// 添加 一 个 角色 

/// </summary> 

/// <param name="role"> 要 添加 的 角色 </param> 
/// <returns> 添 加 的 角色 数 </returns> 

public static int add(UserRole role) 


} 

/// <summary> 

/// 根 据 ID 删除 一 个 角色 

/// </summary> 

/// <param name="id"> 要 删除 的 角色 id</param> 
/// <returns> 删 除 的 角色 数 </returns> 

public static int delete (string id) 

{ 


return EntityUtility.add<UserRightContext, UserRole>(role); 


return EntityUtility.delete (getQuery(), m => m.ID == id); 


} 
// 更 新 角色 信息 
public static int update (UserRole role) 
{ 
return EntityUtility.update<UserRightContext, UserRole>(role); 
} 
/// <summary> 
/// 根 据 ID 得 到 角色 数据 
/// </summary> 
/// <param name="id"> 要 查询 的 角色 id</param> 
/// <returns> 查 询 到 的 角色 数据 </returns> 
public static UserRole getByID(string id) 


} 

/// <summary> 

/// 分 页 得 到 角色 列表 

/// </summary> 

/// <param name="page"> 分 页 参数 </param> 

/// <returns> 得 到 的 指定 页 码 的 角色 列表 </returns> 

public static List<UserRole> getAll (PageDataArgument page) 
{ 


return EntityUtility.selectOne (getQuery(), m => m.ID == id); 


return EntityUtility.selectMany (getQuery(), m => m.ID, page); 


上 
// 得 到 用 于 执行 角色 数据 访问 的 ObjectQuery 对 象 
private static System.Data.Objects.ObjectQuery<UserRole> getQuery() 
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return new SJL.Entity.UserRightContext() .UserRole; 
} 
(2) 在 业务 逻辑 层 SJL.BI1l 中 添加 用 于 角色 管理 的 类 UserRoleBll， 由 于 此 处 没有 业务 
逻辑 ， 只 需 调 用 数据 访问 层 的 相应 方法 即 可 。 
名 提示: 通用 权限 管理 模块 为 本 书 第 1 个 实用 案例 ， 故 给 出 了 完整 代码 。 为 节省 篇 幅 ， 在 
以 后 的 案例 中 ， 如 果 业 务 逻辑 层 只 是 调用 数据 访问 层 相 应 方法 ， 本 身 没有 实现 其 
他 功能 ， 则 书 中 不 再 给 出 业务 逻辑 层 代 码 ， 读 者 可 从 本 书 配 套 光 盘 中 得 到 完整 
代码 。 


public static class UesrRoleBLL 
| 
public static int add(UserRole role) 
{ 
return Dal0.add(role) 7 


} 
public static int delete (string id) 
{ 
return Dal0.delete(id) 
} 
public static int update (UserRole role) 
f 


} 
public static UserRole getByID(string id) 
{ 


return Dal0.update (role); 


return Dal0.getByID (id); 


} 
public static List<UserRole> getA]ll]l (SJL.Common.PageDataArgument page) 
{ 
return Dal0.getAll (page); 
} 
} 


(3) 在 表现 层 中 添加 一 个 ASPNET 页 面 UserRolePage.aspx 作为 表现 层 ， 在 页 面 顶部 
放 一 个 GridView 以 显示 角色 列表 ， 页 面 底部 放 几 个 TextBox 以 允许 用 户 编辑 和 添加 角色 ， 
页 面 代 码 如 下 : 


<asp:Content ID="Content2" ContentPlaceHolderID="ContentPlaceHolder1l" 
runat="server"> 
<h3> 用 户 角 色 管 理 </h3> 
<$-- 在 下 面 这 个 Panel 中 显示 角色 列表 --%> 
<asp:Panel runat="server" ID="listPanel"> 
<asp:GridView ID="GridViewl" runat="server" AutoGenerateColumns= 
"False" 
DataKeyNames="ID" 
CssClass="gridview" onrowcommand="GridView]l RowCommand" > 
<Columns> 
<asp:BoundField DataField="ID" HeaderText=" 角 色 编 码 " SortExpres- 
sion="ID"> 
<HeaderStyle Width="100" /> 
</asp:BoundField> 
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<asp:BoundField DataField="Name" HeaderText=" 角 色 名 称 " SortExpres 
sion="Name"> 
<HeaderStyle Width="120" /></asp:BoundField> 
<asp:BoundField DataField="description"” HeaderText=" 角 色 描 述 " 
SortExpression="Name"/> 
<&%- -编辑 和 删除 为 模板 列 ， 显 示 一 幅 图 片 ， 并 引发 服务 器 端 命令 --%> 
<asp:TemplateField HeaderText=" 编 辑 "> 
<ItemTemplate> 
<asp:ImageButton ID="ImageButtonl" runat="server" Command-— 
Name="edit0" CommandArgument="<%#Eval ("id") %>' ImageUr]l= 
"../images/edit.png" /> 
</ItemTemplate> 
</asp:TemplateField> 
<asp:TemplateField HeaderText=" 删 除 "> 
<ItemTemplate> 
<asp:ImageButton ID="ImageButton2" runat="server" Command 一 
Name="delete0" OnClientClick="return confirmDelete();" 
CommandArgument="'<%#Eval ("id") $%>' ImageUrl="../images/ 
delete.png" /> 
</ItemTemplate> 
</asp:TemplateField> 
</Columns> 
</asp:GridView> 
<webdiyer:AspNetPager ID="pagerl" runat="server" 
onpagechanged="pager1l PageChanged"> 
</webdiyer:AspNetPager> 


<br /> 
<asp:Button ID="addButton" runat="server"” Text=" 添 加 " 
onclick="addButton Click" ><br /> 


</asp:Panel> 
<s%-- 下 面 这 个 Panel 用 于 角色 编辑 --%> 
<asp:Panel ID="editPanel" runat="server" > 
<table> 
<tr><td> 角 色 编 码 </td><td> 
<asp:TextBox ID="moduleID" runat="server"></asp:TextBox> 
<asp:HiddenField ID="hiddenID" runat="server" /> 
</td></tr> 
<tr><td> 角 色 名 称 </td><td> 
<asp:TextBox ID="moduleName" runat="server"></asp:TextBox></td> 
<AEE> 
<tr><td> 角 色 描 述 </td><td> 
<asp:TextBox ID="description" runat="server"></asp:TextBox> 
</td></tr> 
<tr><td colspan="2"> 
<asp:Button ID="saveButton" runat="server"” Text=" 保 存 " 
onclick="saveButton Click" /> 
<asp:Button ID="cancelButton" runat="server" 
Text=" 取 消 " onclick="cancelButton Click" /> 
</td></tr> 
</table> 
</asp:Panel> 


UserRolePage.aspx 页 面 设计 外 观 如 图 11.5 所 示 。 
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图 11.5 角色 管理 设计 界面 


(4) 在 UserRolePage.aspx 页 面 中 添加 JavaScript 代码 ， 实 现 GridView 鼠标 光 棒 效果 、 
删除 提示 、 输 入 验证 等 功能 。 


<g%-- 导 入 CSS 文件 和 JavaScript 文件 --%> 

<link href="../css/ui-lightness/Jjquery-ui-1.7.2.css" rel="stylesheet" 
type="text/css" /> 

<script src="../js/jquery-1.3.2.js" type="text/javascript"></script> 


<script src 


/js/jquery-ui-1.7.2.js" type="text/javascript"></script> 


<script src="../js/MyUtility.j]s" type="text/javascript"></script> 
ip "../js/ ility.js" type=" | ipt"></scrip 
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<script type="text/javascript"> 


//jQuery 初始 化 函数 
$(function () { 


// 为 GridView 实现 光 棒 效果 
$('table.gridview') .find("tr") .hover( 
function() { $(this) .addClass ('hoverRow'); }, 

// 鼠 标 进入 时 添加 hoverRow 样式 
function () { $(this) .removeClass('hoverRow'); } 

// 鼠 标 移出 时 移 除 hoverRow 样式 
); //$('table') .tr.hover 


$SjlUtility.addButtonClass (); // 为 所 有 按钮 应 用 外 观 样式 
$('#<%=saveButton.ClientID%>') .click (checkInput); 


// 为 保存 按钮 添加 输入 验证 
}); //$ (function) 
// 删 除 确认 函数 
function confirmDelete() { return confirm(" 确 实 要 删除 吗 ? "); } 
// 检 测 用 户 输入 函数 


function checkInput () { 


if (checkInput2 () ) 
return true; 
alert (' 必 须 输 入 完整 的 编码 和 名 称 。"' ) ; 


return false; 


function checkInput2() { 


// 获 取 角 色 ID 控件 和 角色 名 称 控件 的 客户 端 ID 
Var userid = "'#<%=roleID.ClientID %>'; 
Var userName = '#<%=roleName.ClientID %>"'; 


// 获 取 角 色 ID 控件 和 角色 名 称 控件 的 文本 
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var a=$ (userid) .val() ; 
if (a == "") return falses 
Var b=$ (userName) .val (); 
IE (Ds= "return Ealser 
return true; 
} 
</script> 


(5) 在 UserRolePage.aspx 页 面 Page Load 事件 中 ， 显 示 第 1 页 角色 列表 。 


protected void Page Load (object sender, EventArgs e) 


if (!IsPostBack) 
{ 
bindRoles () 
displyMode () 
} 


// 绑 定 一 页 角色 数据 到 GridView 
Private void bindRoles () 
{ 
// 生 成 分 页 参数 
PageDataArgument arg = new PageDataArgument (pager1l.CurrentPageInqex — 
1, pagerl.PageSize, true); 


var list = UserRoleBLL.getAll (arg); // 执 行 查询 
pagerl .RecordCount = arg.count; // 刷 新 总 数量 
GridView1.DataSource = list; // 数 据 绑 定 


GridViewl .DataBind(); 
’ 
// 进 入 只 读 模式 ， 隐 藏 编辑 区 域 
private void displyMode() 
{ 

editPanel .Visible 

listPanel .Visible 


false; 
true; 


(6) 当 在 角色 列表 中 单 击 “编辑 ”按钮 或 者 单 击 “ 添 加 ”按钮 时 ， 则 显示 编辑 区 域 。 
如 果 编 辑 已 有 角色 ， 则 将 角色 信息 显示 在 编辑 区 域 ， 如果 添加 新 角色 ， 则 清空 编辑 区 域内 
容 。 页 面 中 定义 了 一 个 string 类 型 的 字段 newID 来 标识 是 否 有 新 记录 。 编 辑 、 删 除 和 添加 
按钮 的 代码 如 下 : 


private const string newID = "*NEWID*"; 
protected void GridViewl RowCommand (object sender, 
GridViewCommandEventArgs e) 
string id = e.CommandArgument.ToString (); // 得 到 角色 ID 
// 如 果 是 删除 命令 ， 则 删除 当前 角色 ， 重 新 绑 定 数据 
if (e.CommandName == "delete0") 
UserRoleBLL.delete(id) 
bindRoles () 
} 
/ /如果 是 编辑 命令 ， 则 切换 到 编辑 模式 
else if (e.CommandName == "edit0") 
{ 
editMode (id); 
} 


~ 
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// 当 单 击 “ 添 加 ”按钮 时 ， 切 换 到 编辑 模式 

protected void addButton Click(object sender, EventArgs e) 

| 
editMode (newID); 

} 

/// <summary> 

/// 进入 编辑 模式 ， 显 示 编辑 区 域 

/// </summary> 

/// <param name="id"> 要 编辑 的 角色 id， 如 果 为 newID， 则 为 添加 新 角色 </param> 

private void editMode (string id) 

{ 
editPanel .Visible 
listPanel .Visible 
hiddenID.Value = id; 
if (id == newID) 
{ 


true; 
false; 


Ul 


roleID.Readonly = false; // 如 果 为 新 增 记录 ， 则 ID 可 修改 
clearEdit (); 

} 

else 

{ 
roleID.Readonly = true; // 如 果 编 辑 原 有 角色 ， 则 ID 只 读 


var module = UserRoleBLL.getByID(id); 
roleID.Text = module.ID; 

roleName.Text = module.Name; 
description.Text = module.Description; 


1 


(7) 当 用 户 单 击 “保存 ”按钮 时 ， 需 要 根据 当前 编辑 的 是 新 数据 还 是 原 有 数据 ， 相 应 
地 对 数据 库 执行 插入 或 者 更 新 操作 。 保 存 按钮 代码 如 下 : 


protected void saveButton Click(object sender, EventArgs e) 
UserRole role = new UserRole(); 
role.Description = description.Text; 
role.ID = rolelID.Text; 
role.Name = roleName.Text; 


if (hiddenID.Value == newID) // 如 果 ID 为 newID， 则 添加 记录 
UserRoleBLL.add (role); 
else // 否 则 更 新 记录 


UserRoleBLL.update (role); 
bindRoles (); 
displyMode (); 
3 


(8) 在 分 页 控件 的 PageIndexChanged 事件 中 ， 重 新 绑 定 数据 。 
protected void pager1l_PageChanged (object sender, EventArgs e) 
| 


bindRoles (); 
} 


(9) 运行 UserRolePage.aspx 页 面 ， 运 行 界面 如 图 11.6 所 示 。 


11.3.2 用户 管理 


户 管理 页 面 可 以 实现 用 户 的 浏览 、 添 加 、 删 除 、 修 改 操作 ， 但 是 不 能 修改 其 他 用 户 
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的 登录 密码 。 实 现 用 户 管理 功能 步骤 如 下 。 


EE ox 
文件 四。 电 和 加、 查看 外 历史 名 ) 书 答 加 “工具 中， 下 助 如 

[< (0 lr EC 日 2 £ 
Google ] 基本 M- 合 安 - 稳 -到 + © -@- 
口 wttp:/7lecalkes 一 raelepage- aspx 画 加 


etnies 用 户 角色 管理 

ct 角色 编 双 角色 名 称 角色 描述 EEC3 

功能 模 失 管理 ol 系统 得 只 具有 最 高 权限 ， 不 要 但 除 焉 修改 此 角色 了 了 XX 

有 o2 测试 02s 测试 用 的 Hx 

个 允 三 有 

质 析 所 有 ， 种 航天 。 枝 椒 云 竺 ;sun j. 1. studic8email com 

ae EY 
EE [大 加 4 


图 11.6 角色 管理 运行 界面 


(1) 在 数据 访问 层 SIL.Dal 中 添加 一 个 类 UserDal， 并 在 其 中 实现 对 数据 库 中 User 表 
的 增 、 删 、 改 查 功 能 。 


public class UserDal 


| 
// 添 加 用 户 
public static int add(User user) 


{ 


} 

/7 删除 用 户 

public static int delete(string id) 
{ 


return EntityUtility.add<UserRightContext, User> (user); 


return EntityUtility.delete (getQuery(), m => m.ID == id); 
} 
/// <summary> 
/// 更 新 用 户 信息 ， 但 不 更 新 密码 
/// </summary> 
/// <param name="user"> 要 更 新 的 用 户 信 息 </param> 
/// <returns> 被 更 新 的 用 户 数 </returns> 
public static int update (User user) 
{ 
return EntityUtility.update<UserRightContext, User> (user, 
"Password"); 


} 
// 根 据 用 户 ID 得 到 用 户 信 息 
public static User getByID (string id) 
{ 
return EntityUtility.selectOne (getQuery(), m => m.ID == id); 
} 
/// <summary> 
/// 分 页 获取 用 户 列表 
/// </summary> 
/// <param name="page"> 分 页 参数 </param> 
/// <returns> 指 定 页 面 用 户 列表 </returns> 
public static List<User> getAll]l (PageDataArgument page) 
{ 


= 
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Var query = getQuery(); 

Var list= EntityUtility.selectMany (query, m => m.ID, page,false); 
// 加 载 用 户 列表 

list.ForEach(u => u.UserRoleReference.Load() ) ; // 加 载 用 户 角 色 

query.Context .Dispose(); 

return list; 


上 
public static bool isAdmin (User user) 


: 


return User-RoleID == "01"; 


' 


public static int changePassword(User user) 


{ 


} 
private static System.Data.Objects.ObjectQuery<User> getQuery () 


{ 


return EntityUtility.update<UserRightContext, User> (user); 


return new UserRightContext () .User; 


(2) 在 业务 逻辑 层 SIL.Bll 中 添加 一 个 UserBll 类 ， 实 现 与 用 户 数据 相关 的 业务 规则 。 


EE 要 


有 两 条 业务 规则 : 一 是 系统 管理 员 用 户 admin 不 能 被 删除 ， 二 是 添加 新 用 户 时 设置 默 


认 密 码 123456。UserBll 类 的 关键 代码 如 下 : 


public static class UserBLL 


{ 


) 


Private const string ConstDefaultPassword = "123456"; 
public static int add(User user) 
{ 
user.Password = ConstDefaultPassword; // 添 加 用 户 时 设置 默认 密码 
return UserDal.add (user); 
} 
public static int delete (string id) 
{ 
// 如 果 要 删除 admin， 则 不 允许 ， 给 出 错误 提示 
if (id.ToLower() == "admin") 
throw new ApplicationException ("系统 管理 员 用 户 admin 不 能 被 删除 ") ; 
return UserDal.delete(id); 


(3) 在 表现 层 SJL.Web 项 目 中 添加 一 个 用 户 管理 页 面 ManageUser.aspx。 页 面 布局 与 
角色 管理 页 面 类 似 ， 页 面 代码 如 下 : 


<h3> 系 统 用 户 管理 </h3> 
<asp:Panel runat="server" ID="1istPanel"> <s-- 用 户 信息 显示 区 域 --%> 


<asp:GridView I 
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GridViewl" runat="server" AutoGenerateColumns="False" 
DataKeyNames="ID" 
CssClass="gridview" onrowcommand="GridViewl RowCommand" > 
<Columns> 
<asp:BoundField DataField="ID"” HeaderText=" 用 户 编码 ”> 
<HeaderStyle Width="100px" /> 
</asp:BoundField> 
<asp:BoundField DataField="UserName" HeaderText=" 用 户 名 称 "” > 
<HeaderStyle Width="120px" /> 
</asp:BoundField> 
<asp:BoundField DataField="Password" HeaderText=" 密 码 " 
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Visible="False" /> 
<asp:TemplateField HeaderText=" 用 户 角色 "> 
<%-- 模 板 列 ， 显 示 用 户 角色 名 称 --%> 

<ItemTemplate> 
<%$#Eval ("UserRole.Name")®> 
</ItemTemplate> 
</asp:TemplateField> 
<&%-- 模 板 列 ， 以 图 形 方式 显示 编辑 按钮 --$> 
<asp:TemplateField HeaderText=" 编 辑 "> 
<ItemTemplate> 
<asp:ImageButton runat="server" CommandName="edit0" Command- 
Argument="<%#Eval ("id") %>' ImageUrl="../images/edit.png" /> 
</ItemTemplate> 
</asp:TemplateField> 
<&%- -模板 列 ， 以 图 形 方式 显示 删除 按钮 --$> 
<asp:TemplateField HeaderText=" 删 除 "> 
<ItemTemplate> 
<asp:ImageButton runat="server" CommandName="delete0" OnClient-— 
Click="return confirmDelete();"CommandArgument="'<%#Eval ("id") $%>' 
ImageUrl="../images/delete.png" /> 
</ItemTemplate> 
</asp:TemplateField> 

</Columns> 

</asp:GridView> 

<webdiyer:AspNetPager ID="pagerl" runat="server" 
onpagechanged="pager1l PageChanged"> 

</webdiyer:AspNetPager> 


<br /> 
<asp:Button ID="addButton" runat="server"” Text=" 添 加 " 
onclick="addButton Click" /><br /> 


</asp:Panel> 
<asp:Panel ID="editPanel" runat="server" > <s-- 用 户 信息 编辑 区 域 --%> 
<table> 
<tr><td> 用 户 编码 </td><td> 
<asp:TextBox ID="userid" runat="server"></asp:TextBox> 
<asp:HiddenField ID="hiddenID" runat="server" /> 
</td></tr> 
<tr><td> 用 户 名 称 </td><td> 
<asp:TextBox ID="userName" runat="server"></asp:TextBox></td></tr> 
<tr><td> 用 户 角色 </td><td> 
<asp:DropDownList ID="roleList" runat="server" DataTextField="name" 
DataValueField="id"> 
</asp:DropDownList> 
</td></tr> 
<tr><td colspan: 
<asp:Button ID="saveButton" runat="server" Text=" 保 存 " 
onclick="saveButton Click" /> 
<asp:Button ID="cancelButton" runat="server" 
Text=" 取 消 " onclick="cancelButton Clickl" /> 
</td></tr> 
</table> 
</asp:Panel> 


(4) 在 ManageUseraspx 页 面 的 Page Load 事件 中 ， 绑 定 GridView 用 户 列 表 和 
DropDownList 角色 列表 。 
protected void Page Load (object sender, EventArgs e) 


- 
if (!IsPostBack) 


30s 


initData() 7 
displyMode (); 


} 

// 初 始 化 数据 ， 包 括 绑 定 用 户 列表 、 角 色 列 表 
private void initData() 

. 


bindUsers (); 


// 获 取 和 角色 列表 并 绑 定 到 DropDownList 控件 中 


Var roles = UserRoleBLL.getRAll (SJL.Common.PageDataArgument .allData); 


roleList.DataSource = roles; 
roleList.DataBind(); 
// 在 下 拉 表 中 添加 一 个 “请 选择 ”的 项 


roleList.Items.Insert (0,new ListItem("-- 请 选择 --"，"-1")); 


} 
// 绑 定 用 户 列表 
private void bindUsers () 


{ 


PageDataArgument arg = new PageDataArgument (pagerl .CurrentPageIndex — 


1, pagerl.PageSize, true); 
Var list = UserBLL.getAll (arg); 
pagerl .RecordCount = arg.count; 
GridViewl .DataSource = list; 
GridViewl .DataBind(); 

” 


(5) ManageUseraspx 页 面 上 编辑 、 删 除 、 添 加 、 保 存 按钮 的 代码 与 角色 管理 页 面 
UserRolePage.aspx 中 的 相应 代码 类 似 ， 此 处 不 再 给 出 具体 代码 。 
(6) 运行 ManageUseraspx 页 面 ， 运 行 界面 如 图 11.7 所 示 。 
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图 11.7 用 户 管理 页 面 


11.3.3 ”功能 模块 管理 


功能 模块 是 权限 分 配 的 基本 单位 , 在 ASPNET 应 用 程序 中 , 一 个 功能 模块 通常 对 应 于 


一 个 ASPNET 页 面 。 
的 具体 实现 与 前 面 所 介绍 的 角色 管理 基本 相同 。 当 Web 应 月 


个 添加 这 些 页 面 ， 操 
生成 页 面 数据 的 方法 ， 代 码 如 下 : 


* 


日 中 有 许多 页 面 时 ， 如果 


用 户 需 要 对 功能 模块 进行 浏览 、 添 加 、 删 除 、 修 改 等 操作 ， 这 些 功 能 
户 逐 
E 将 非常 繁琐 ， 为 此 ， 可 以 编写 一 个 根据 网 站 目录 下 的 aspx 文件 自动 
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public static class AdminTool 


| 


// 根 据 网 站 中 的 页 面 自动 生成 模块 信息 
public static void genereateModules () 
string path = HttpContext.Current -Server.MapPath ("~"); 
generate( new DirectoryInfo (path) ) 
} 
static int n= 1; 
/// <summary> 
/// 递归 方法 ， 根 据 目 录 下 的 aspx 文件 生成 模块 信息 
/// </summary> 
/// <param name="directory"> 路 径 </param> 
private static void generate (DirectoryInfo directory) 
{ 
FileInfo[] files = directory.GetFiles("*.aspx"); 
// 得 到 目录 下 的 所 有 aspx 文件 
foreach (var item in files) 
{ 
// 如 果 数 据 库 不 存在 此 页 面 ， 则 添加 
if (ApplicationModuleBLL.getByUr]l (item.Name) == null) 
{ 
ApplicationModule m = new ApplicationModule(); 
m.ID = string.Format ("{0:00}", n++); 
m.Name = "自动 生成 的 模块 "; 
m.Description = "自动 生成 的 页 面 ， 没 有 描述 。"; 
m.URL = item.Name; 
ApplicationModuleBLL.add (m); 
} 
} 


DirectoryInfo[] dirs = directory.GetDirectories(); 
foreach (var item in dirs) 


{ 
generate (item); // 递 归 进 入 下 一 级 目录 
} 


11.3.4 角色 权限 管理 


恒 


前 面 几 个 阶段 建立 了 通用 权限 管理 模块 的 基础 数据 ， 接 下 来 需要 为 不 同 的 角色 分 配 权 


具体 步骤 如 下 : 
(1) 在 数据 库 中 添加 一 个 存储 过 程 以 刷新 角色 权限 矩阵 。 


CREATE PROCEDURE [dbo] . [sp_refreshRoleRight] 
AS 
BEGIN 


set nocount on; 


一 -删除 非法 的 角色 数据 


delete from roleright where roleid not in(select id from userrole); 


=-- 删 除非 法 的 模块 数据 
delete from roleright where moduleid not in(select id from applica— 
七 Ionmodule) 


=-- 剧 新 角色 权限 矩阵 
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insert into roleright (roleid,moduleid, theright) 
select rAd ,meld "0 
from userrole r Ccross join applicationmodule mm 
where not exists 
(select * from roleright rr where rr.roleid=r.id and rr.moduleid=m.id ) 
END 


(2) 在 数据 访问 层 SIL.Dal 中 添加 一 个 类 RoleRightDal， 实 现 对 角色 权限 的 增 、 删 、 
改 、 查 功能 ， 关 键 代码 如 下 : 


public static class RoleRightDal 

{ 
// 添 加 角色 权限 
public static int add(RoleRight right) 
{ 


} 

// 更 新 角色 权限 

public static int update (RoleRight right) 
{ 


return EntityUtility.add<UserRightContext, RoleRight> (right); 


return EntityUtility.update<UserRightContext, RoleRight> (right); 
} 
/ /根据 角色 和 模块 获取 权限 信息 
public static RoleRight getByID(string role, string module) 
{ 
Var i= EntityUtility.selectOne (getQuery(), m => m.RoleID == role && 
m.ModuleID == module); 
//1loadDetail (i); 
return i; 


} 
// 获 取 某 角色 的 所 有 权限 信息 
// 参 数 : role 角色 ID， ”page 分 页 参数 
public static List<RoleRight> getBYRole (string role, PageDataArgument 
page) 
{ 
var list= EntityUtility.selectMany (getQuery(), r =>r.ModuleID, page, 
m => m.RoleID == role); 
//list.ForEach(i => loadDetail (i)); 
return list; 
} 
// 刷 新 角色 权限 数据 
public static void refreshRoleRight () 
{ 
Var context = new UserRightContext (); 
context .refreshRoleRight (); 
context .Dispose(); 
} 
/// <summary> 
/// 用 户 是 否 有 访问 指定 页 面 的 权限 
/// </summary> 
/// <param name="role"> 角 色 ID</param> 
/// <param name="page"> 请 求 访问 的 页 面 </param> 
/// <returns> 是 否 有 权限 </returns> 
public static bool canAccessPage (string role, string page) 
{ 
// 根 据 URL 获取 对 应 的 模块 数据 
string module=ApplicationModuleDal .getByUr] (page) .ID; 
// 获 取 角 色 对 该 模板 的 权限 ， 并 判断 是 否 可 以 访问 
var right = EntityUtility.selectOne (getQuery(), r => 工 -RoleID == role 
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&& -ModuleID == module); 


| 


return right.TheRight == "1"7 


private static System.Data.Objects.ObjectQuery<RoleRight> getQuery () 


{ 


return new UserRightContext () .RoleRight; 


(3) 在 业务 逻辑 层 SIL.Bll 中 添加 一 个 类 RoleRightBll。 由 于 本 例 业务 规则 简单 ， 此 类 
的 主要 功能 为 向 表现 层 提供 接口 ， 并 调用 数据 访问 层 相应 方法 ， 为 节省 篇 幅 ， 此 处 省 略 
RoleRightBll 类 的 代码 。 

(4) 在 表现 层 SJL.Web 中 添加 一 个 RoleRightPage.aspx 页 面 ， 用 于 实现 权限 分 配 界面 。 
页 面 用 GridView 列 出 了 当前 权限 分 配 状态 ， 并 可 以 通过 单 击 “ 人 允许” 和 “拒绝 ”按钮 进行 


权限 变更 。 


<h3> 角 色 权 限 管理 </h3> 

选择 角色 : <asp:DropDownList ID="roleList" runat="server" DataTextField= 
"name" DataValueField="id"> 

</asp:DropDownList> 


<asp:Button ID="okButton" runat: 


server"” Text=" 确 定 " onclick="okButton_ 


Click" /> <asp:HiddenField ID="hiddenRole" runat="server" /><br /><br /> 
<$ 一 -显示 角色 权限 分 配 列表 --%> 
<asp:GridView ID="GridViewl" runat="server" AutoGenerateColumns="False" 
CssClass="gridview" onrowcommand="GridViewl RowCommand" > 
<Columns> 


<asp:TemplateField HeaderText=" 模 块 编码 "> 


<ItemTemplate> 
<asp:Label ID="Labell" runat="server" 
Text="'<%# Eval ("ApplicationModule.id") $%>'></asp: 
Label> 
</ItemTemplate> 
<ItemStyle Width="80px" /> 
</asp:TemplateField> 
<asp:TemplateField HeaderText=" 模 块 名 称 "> 
<ItemTemplate> 
<asp:Label ID="Label2" runat="server" 
Text="'<%# Eval ("ApplicationModule.name") $%>'></asp:Label> 
</ItemTemplate> 
<ItemStyle Width="100px" /> 
</asp:TemplateField> 
<asp:TemplateField HeaderText=" 模 块 描述 "> 
<ItemTemplate> 
<asp:Label ID="Label3" runat="server" 
Text="'<%# Eval ("ApplicationModule.description") $%$>'> 
</asp:Label> 
</ItemTemplate> 
<ItemStyle Width="200px" /> 
</asp:TemplateField> 
<asp:TemplateField HeaderText=" 有 无 权限 "> 
<ItemTemplate> 
<asp:CheckBox ID="CheckBoxl" runat="server" 
Checked="'<%#Eval ("theright") .ToString() .Trim()=="1" %$>"' 
Enabled="false" /> 
</ItemTemplate> 
</asp:TemplateField> 
<asp:TemplateField HeaderText=" 操 作 "> 


= 


第 3 篇 ”项目 实战 


<ItemTemplate> 
<asp:LinkButton runat="server" ID="grant" Text=" 人 允许 " Command- 
Name="grant" CommandArgument="'<%#Eval ("ApplicationModule. 
fe bed) E /> 
<asp:LinkButton runat="server" ID="deny" Text=" 禁 止 " Command- 
Name="deny" CommandArgument="'<%#Eval ("ApplicationModule. 
id")%>"/> 
</ItemTemplate> 
</asp:TemplateField> 
</Columns> 
</asp:GridView> 
<webdiyer:AspNetPager ID="pagerl" runat="server" 
onpagechanged="pagerl PageChanged"> 
</webdiyer:AspNetPager> 


RoleRightPage.aspx 页 面 设计 外 观 如 图 11.8 所 示 。 


UserRightMaster.master 
Conkertplacetolerl ( 自 定义 ) Ln EE 
系统 用 户 管理 
用 户 角色 管理 丑 择 角色 ，[ 采 于 定 了 ] 确定 
功 甬 模 块 管理 NiddenField - hiddenkole 
角色 权限 管理 
修改 窜 码 模块 编 到 模块 名 称 异 块 摘 述 有 无 权限 。 操作 
数据 效 定 数据 将 定 数据 昂 定 [| 光 许 禁止 
数据 闭 定 数据 独 定 数据 儿 定 La 允许 禁止 
数据 效 定 数据 绑 定 数据 绑 定 请 区 许 禁止 
数据 贸 定 。。 数据 绑 定 效 据 绑 定 [= 思 许 蔡 止 
数据 比 定 数据 独 定 数据 郑 定 口 区 许 禁止 
kK¢¢12345678910...>>> 
Sur JT sud oni om 
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图 11.8 角色 权限 管理 设计 界面 


(5) 在 页 面 的 Page_Load 事件 中 ， 将 角色 数据 绑 定 到 DropDownList 控件 。 由 于 系统 
管理 员 角 色 拥 有 全 部 权限 ， 所 以 不 需要 为 其 分 配 权限 ， 从 而 在 下 拉 列 表 中 不 需要 显示 系统 
管理 员 角 色 。 


protected void Page Load (object sender, EventArgs e) 
{ 
if (!IsPostBack) 


{ 
RoleRightBLL .refreshRoleRight (); 


initRoles (); 
} 
y 
private void initRoles () 
{ 
var list = UserRoleBLL.getAll]l (PageDataArgument .allData); 
// 得 到 所 有 角色 列表 
// 系 统管 理 员 具有 全 部 权限 ， 不 需要 分 配 ， 所 以 从 列表 中 删除 系统 管理 员 角 色 
11st-Remove (list.Find(r => r.ID == "01")); 
roleList.DataSource = list; 
roleList.DataBind(); 
if (roleList.Items.Count == 0) 


{ 
ClientScript.RegisterSstartupScript (this.GetType (), 
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"addrolefirst", 


"<script> 数 据 库 中 没有 角色 。 请 先 添加 角色 再 为 其 分 配 权 限 。</script>"); 


return; 
} 
roleList.SelectedIndex = 0; 
bindRights (); 


(6) 在 “确定 ”按钮 的 Click 事件 中 ， 根 据 用 户 所 选择 的 角色 列 出 角色 权限 分 配 情况 。 


// 显 示 角 色 权 限 列表 

private void bindRights () 

| 
if (roleList.SelectedIndex < 0) return; // 如 果 未 选择 角色 则 返回 
string role = roleList.SelectedValue; // 得 到 角色 ID 
hiddenRole.Value = role; 
/ /创建 分 页 查询 参数 


PageDataArgument page = new PageDataArgument (); 
page.refreshCount = true; 
page.pageSize = pagerl .PageSize; 
page.pageIndex = pagerl.CurrentPageIndex - 1; 
// 执 行 查询 ， 将 查询 结果 绑 定 到 GridView 
var list = RoleRightBLL.getBYRole (role,page); 
pagerl .RecordCount = page.count; 
GridViewl .DataSource = list; 
GridViewl .DataBind(); 

} 

protected void okButton Click(object sender, EventArgs e) 
pagerl .CurrentPageIndex = 1; 
bindRights(); 

} 


(7) 当 用 户 单 击 列表 中 的 “允许 ”或 “禁止 ”按钮 时 , 会 触发 GridView 的 RowCommand 


事件 ， 需 要 在 此 事件 中 执行 相应 的 权限 改动 ， 代 码 如 下 : 


protected void GridViewl RowCommand (object sender, GridViewCommandEvent— 


Args e) 
{ 
string newRigth="0"; 
// 根 据 用 户 选择 的 是 允许 还 是 禁止 ， 设 置 新 权限 的 值 


if (e.CommandName == "grant") 
newRigth = "1"; 
else if (e.CommandName == "deny") 
newRigth = "0"; 
else 
return; 
string id = e.CommandArgument .ToString (); 
// 更 新 权限 数据 


RoleRight right = new RoleRight() 
right.ModuleID = id; 
right.TheRight = newRigth; 
right.RoleID = hiddenRole.Value; 
RoleRightBLL.update (right); 


bindRights (); // 重 新 绑 定 权限 列表 


3 
(8) 运行 RoleRightPage.aspx 页 面 ， 运 行 界 面 如 图 11.9 所 示 。 
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文件 如 编辑) 查看 轨 。 历史 GE) 书签 外 工具 上 帮助 0 
鲍 林 Coho oarb/ 7 - 
Ca 2 OE :> i i 


| http://localhos™RightPage. aspz| 7 


一 选择 角色 : [Ko2* 加 确定 


模块 编码 。。 模块 名 称 模块 描述 有 无 权限 。 操作 

iol 音 录 自动 生成 的 页 面 ， 没 有 拱 述 。 历久 许 蔡 目 

-io 首页 自动 生成 的 页 面 ， 没 有 措 述 。 网 记 许 葬 目 

03 用 于 纺 程 测 这 的 自动 生 成 的 页 面 ， 没 有 描述。 区 爷 许 酌 目 

04 售 改 密码 自动 生成 的 页 面 ， 没 有 描述。 局 允许 禁止 
自动 生成 的 页 面 ， 没 有 拱 述 。 加 


图 11.9 权限 分 配 页 面 


11.4 权限 控制 


本 节 将 介绍 通用 权限 管理 系统 核心 功能 的 实现 ， 即 权限 检测 与 控制 。 在 具有 权限 管理 
功能 的 ASPNET Web 应 用 中 ，, 当 用 户 要 访问 某 一 个 页 面 时 , 首先 判断 此 用 户 是 否 拥有 访问 
所 请 求 页 面 的 权限 ， 如 果 有 则 继续 访问 ， 如 果 没 有 则 提示 权限 不 足 。 可 以 在 页 面 的 
Page_Load 事件 中 对 用 户 权限 进行 检测 ,但 是 由 于 一 个 ASPNET 应 用 程序 中 会 有 多 个 页 面 ， 
这 种 方案 需要 在 每 个 页 面 中 都 编写 代码 ， 非 常 不 方便 。 这 个 问题 还 存在 一 种 更 好 的 解决 方 
案 ， 即 HTTP 模块 (HttpModule)。 


11.4.1 用 户 权 限 检 测 


在 ASPNET 应 用 程序 中 ， 每 个 HTTP 请 求 发 生 时 ， 都 会 执行 已 经 注册 的 HttpModule 
相应 方法 。 因 此 , 在 HttpModule 中 适合 编写 所 有 页 面 的 公共 代码 。 在 通用 权限 管理 系统 中 ， 
每 个 页 面 被 请 求 时 ， 都 需要 检测 用 户 权限 是 否 合适 ， 可 以 把 用 户 权限 检测 代码 写 到 
HttpModule 中 。 
进行 用 户 权 限 检测 的 过 程 如 下 :用 户 请 求 一 个 URL( 如 http://localhost/book.aspx?id=2)， 
程序 先 取出 URL 中 包含 的 文件 名 (如 book.aspx)， 文 件 名 通常 为 URL 中 最 后 一 个 “/” 号 
后 面 和 “2?” 号 前 面 的 部 分 ， 然 后 再 判断 此 文件 是 否 需 要 受权 限 保护 。 如 果 受 权限 保护 ， 则 
要 根据 当前 登录 的 用 户 角色 判断 用 户 是 否 拥 有 访问 此 文件 的 权限 。 如 果 权 限 检测 通过 ， 则 
允许 继续 访问 指定 资源 ， 和 否则 跳 转 到 登录 页 面 。 图 11.10 描述 了 权限 检测 流程 。 

实现 用 户 权限 检测 和 控制 的 具体 步骤 和 代码 如 下 : 

(1) 在 表现 层 项 目 SIL.Web 中 添加 一 个 ASPNET 模块 ， 命 名 为 CheckUserModule。 

(2) 在 CheckUserModule 的 Init 事件 中 为 HITP 应 用 的 AcquireRequestState 事件 注册 

个 事件 处 理 程序 。 
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public void Init(HttpApplication context) 
{ 

context.AcquireRequestState += new EventHandler (checkUserRight); 
} 


请 求 一 个 URL 


取得 此 URL 


请 求 文件 为 i 
受 保 六 资源 


用 户 是 否 登录 


用 户 是 否 有 


足够 权限 


图 11.10 权限 检测 流程 


允许 访问 


(3) 在 checkUserRight0 方 法 中 ， 实 现 图 11.10 所 描述 的 检测 流程 。 


/// <summary> 
/// 检测 用 户 权 限 
/// </summary> 
void checkUserRight (object sender, EventArgs e) 
{ 
HttpApplication application = (HttpApplication) sender; // 获 取 应 用 程序 
string url = HttpContext.Current.Request.Url.ToString(); // 获 取 URL 
int start=url.LastIndexOf("/") + 1; // 查 找 URL 中 最 后 一 个 /的 位 置 
int end=url.IndexOf ('?',start); // 查 找 URL 中 ? 位 置 
string requestPage = null; 
if (end < 0) end = url.Length - 1; 
requestPage=url.Substring (start，end - start +1); // 得 到 所 请 求 的 页 面 
requestPage = requestPage.ToLower (); 
if (requestPage == loginPage) return; 
if (!isProtectedResource (requestPage)) return; 
User user=SJL.Web.HttpCode.WebUtility.currentUser; // 获 得 当前 用 户 
// 如 果 当 前 用 户 为 空 ， 则 转 到 登录 页 面 
IE (user==null) 
{ 
application.Response.Redirect ("~/Login.aspx"); 
return; 


上 
// 如 果 当 前 用 户 为 系统 管理 员 ， 则 不 需要 验证 权限 ， 直 接 返 回 
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if (SUL-B11.UserRight.UserBLL.isRAdmin (user)) return; 


// 检 测 用 户 权限 
站 征 (!SJL.B1]1 .UserRight .RoleRightBLL.canAccessPage (user.RoleID, 
requestPage)) 


application.Response.Redirect ("~/AccessDeny.htm"); 
} 
/// <summary> 
/// 判断 页 面 是否 为 受权 限 管理 保护 的 资源 (如 Aspx 等 ) 
/// </summary> 
/// <param name="page"> 所 请 求 的 页 面 </param> 
/// <returns> 是 否 受 保护 </returns> 
bool isProtectedResource (string page) 
page = page.ToLower (); 
string[] protectedFiles = {".aspx",".asmx",".ashx"};// 受 保护 资源 的 扩展 名 
for (int i = 0; i < protectedFiles.Length; i++) 
{ 
if (page.EndsWith (protectedFiles[i])) 
return true; 


} 
ApplicationModule module = SJL.B]1]1.UserRight.ApplicationModuleBLL. 


getByUr] (page); 
if (module == null) return true; 
return !module.IsPublic; // 如 果 页 面 为 公共 模块 则 不 受 保护 


上 述 代码 中 用 到 了 WebUitility 类 的 一 个 属性 currentUser， 该 属性 用 于 读 取 和 设置 当前 


Session 中 保存 的 用 户 信 息 ， 代 码 如 下 : 


public static class WebUtility 
| 


private const string SessionUser = "thisuser"; // 登 录用 户 
public static SJL.Entity.User currentUser 
{ 
get 
{ 
if (HttpContext.Current == null) return null; 
return HttpContext.Current.Session[SessionUser] as global:: 
SJL.Entity.User; 


Se 


{ 


HttpPContext -Current.Session[SessionUser] = value; 


} 


11.4.2 ”用户 登 录 


转 到 登录 页 面 以 合适 的 用 户 身 份 登录 ， 再 访问 此 前 所 请 求 的 资源 。 登 录 页 面 除 了 验证 用 户 


名 和 


在 用 户 权限 检测 模块 中 ， 如 果 用 户 不 具有 访问 所 请 求 资源 的 权限 ， 则 用 户 通 常 需要 跳 


密码 的 正确 性 以 外 ， 还 将 用 户 登录 信息 保存 到 Session 中 ， 以 方便 在 整个 应 用 程序 中 


使 有 


。 登 录 页 面 Login.aspx 布局 和 功能 都 很 简单 ， 代 码 如 下 : 


“Be 
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<div id="login"> 
用 户 名 <asp:TextBox ID="userName" runat="server" ></asp:TextBox> 
密 gnbsp; 码 
<asp:TextBox ID="password" runat="server" TextMode="Password" > 
</asp:TextBox> 
<asp:Button ID="login2" runat="serVer" Text=" 登 录 " Width="80" Height= 
"25" onclick="login2 Click"/> 
</div> 
protected void login2 Click(object sender, EventArgs e) 
{ 
// 获 取 用 户 输入 的 用 户 名 和 密码 
string u = UserName .Text7 
string p = password.Text; 
var user = UserBLL.getByID(u); // 根 据 用 户 名 得 到 用 户 信息 
// 若 密码 正确 ， 则 将 用 户 保存 到 Session， 跳 转 到 首页 


if (user.Password == p) 


WebUtility.currentUser = user; 
Response.Redirect ("~/Default/Default .aspx"); 


} 
// 若 密码 不 正确 ， 则 用 JavaScript 显示 提示 消息 
else 
i 
ClientScript.RegisterStartupScript (this.GetType(), "loginerror", 


"<script>alert (' 用 户 名 或 密码 有 误 ， 登 录 失败 ! ') ;</script>"); 


11.5 小 结 


本 章 设计 开发 了 一 个 ASPNET 通用 权限 管理 系统 ， 可 以 实现 基于 页 面 的 简单 权限 控 
制 。 此 权限 管理 系统 可 以 不 加 修改 地 用 于 各 个 ASPNET 应 用 程序 ,只 需要 在 目标 应 用 程序 
中 注册 HttpModule， 并 添加 相应 的 数据 〈 包 括 用户 、 角 色 、 功 能 模块 、 权 限 ) 即 可 。 


i 


第 12 章 县 长 公开 电话 受理 系统 


本 章 所 介绍 的 案例 是 作者 为 某 县 政府 开发 的 一 个 软件 。 出 于 安全 和 隐私 方面 的 原因 ， 
书 中 不 便 给 出 此 县 的 真实 名 称 ， 下 文 均 以 A 县 代替 。 为 了 更 加 及 时 地 了 解 群众 所 关心 的 问 
题 ， 及 时 为 群众 排忧解难 ，A 县 政府 设立 了 书记 、 县 长 公开 电话 。 为 了 准确 高 效 地 管理 公 
开 电 话 信息 ， 跟 踪 监 控 问 题 处 理 情 况 ， 特 开发 了 本 软件 系统 。 


12.1 整体 设计 思 


本 节 将 介绍 A 县 书记 、 县 长 公开 电话 业务 处 理 流程 ， 分 析 公 开 电话 受理 系统 项 目的 需 
求 ， 提 出 此 系统 整体 设计 思路 ， 实 现 数据 库 结 构 设 计 。 


12.1.1 需求 分 析 


A 县 政府 设立 了 书记 、 县 长 公开 电话 ， 群 众 可 以 随时 拨打 此 电话 反映 问题 。 公 开 电 话 
受理 中 心 有 数 名 工作 人 员 ， 负 责 接 听 群 众 电 话 ， 记 录 群 众 所 反 映 的 问题 ， 并 将 问题 提交 给 
相关 责任 人 。 公 开 电 话 受理 中 心 的 工作 人 员 需 要 跟踪 群众 所 反映 问题 的 整个 处 理 过 程 ， 并 
对 数据 进行 一 定 的 统计 分 析 。 公 开 电 话 业 务 处 理 大 致 流程 如 下 : 

(1) 群众 拨打 公开 电话 ， 受 理 中 心 工作 人 员 接 听 并 记录 群众 所 反映 的 问题 。 在 本 系统 
中 ， 群 众 通过 公开 电话 所 反映 的 问题 称 为 “事件 ”。 

(2) 公开 电话 受理 中 心 人 员 对 群众 所 反映 的 事件 进行 初步 分 析 , 确定 事件 的 重要 级 别 ， 
并 做 出 不 同 处 理 。 如 果 所 反映 问题 较 小 ， 则 受理 中 心 工作 人 员 可 予以 解答 ， 处 理 过 程 到 此 
结束 。 如 果 问 题 较 大 ， 则 需要 根据 问题 反映 的 具体 内 容 及 所 涉及 的 相关 部 门 ， 将 问题 转交 
给 县 领导 或 者 责任 单位 (如 教育 局 、 卫 生 局 、 环 保 局 等 ) 进行 处 理 。 

(3) 群众 所 反映 的 事件 由 公开 电话 受理 中 心 转交 给 县 长 或 者 其 他 责任 单位 后 ， 相 关 责 
任 人 需 要 及 时 解决 问题 ， 并 随时 将 进展 情况 通报 给 公开 电话 受理 中 心 。 

(4) 一 个 事件 处 理 结束 后 ， 公 开 电话 受理 中 心 工作 人 员 需 要 对 当初 反映 此 事件 的 群众 
进行 回访 ， 调 查 群 众 对 于 事件 处 理 结果 的 满意 程度 。 

(5) 公开 电话 受理 中 心 人 员 需 要 对 公开 电话 受理 事件 数据 进行 汇总 ， 形 成 报表 。 

根据 以 上 业务 流程 的 分 析 ， 可 以 确定 公开 电话 受理 系统 主要 有 以 下 需求 : 

口 用 户 管理 和 权限 管理 。 

口 记录 群众 通过 公开 电话 所 反映 的 事件 信息 。 

口 跟踪 公开 电话 所 受理 事件 的 处 理 进展 ， 并 及 时 更 新 导 


件 状 态 。 


ed 
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口 
口 
口 


打印 公开 电话 处 理 过 程 中 产生 的 各 种 单据 。 
按照 各 种 条 件 查询 公开 电话 受理 事件 情况 。 
产生 各 种 统计 报表 。 


12.1.2 ”数据 库 结 构 设 计 


本 系统 最 为 核心 的 数据 为 公开 电话 受理 事件 详情 ,如 受理 时 间 、 反 映 人 、 反 映 内 容 等 。 
与 此 相对 应 , 数据 库 中 用 于 存储 此 数据 的 表 EventInfo 最 为 重要 ,结构 也 较 复杂 ,其 他 表 结 
构 都 较为 简单 。EventInfo 表 中 各 字段 的 含义 如 下 : 


口 


口 
口 
口 


OOODD 


EventID: 事件 编号 ，nvarchar(10) 类 型 ， 主 键 。 

HandleDate: 事件 受理 日 期 ，datetime 类 型 ， 默 认 值 为 当前 日 期 。 

HandleUser: 受理 人 员 ，nvarchar(10) 类 型 。 

EventSource: 事件 来 源 ， 表 示 来 源 于 书记 公开 电话 、 县 长 公开 电话 、 县 长 信箱 等 ， 
int 类 型 ， 外 键 关 联 到 EventSource 表 。 

PersonName: 反映 人 姓名 ，nvarchar(10) 类 型 。 

PersonPhone: 反映 人 联系 电话 ，nvarchar(20) 类 型 。 

PersonAddress: 反映 人 地 址 ，nvarchar(50) 类 型 。 

TypeID: 事件 类 型 表示 所 反映 问题 属于 哪 一 类 (如 环保 、 教 育 、 医 疗 等 ) ，int 
类 型 ， 外 键 关联 到 EventType 表 。 

EventTitle: 事件 主题 ，nvarchar(50) 类 型 。 

EventContent: 事件 内 容 ，nvarchar(1000) 类 型 。 

StatusID: 事件 处 理 状态 ， 如 办 结 、 待 办 等 ，int 类 型 ， 外 键 关联 到 EventStatus 表 。 
EventResult: 事件 处 理 结果 ，nvarchar(1000) 类 型 。 

EvaluationID: 群众 满意 程度 ，int 类 型 ， 外 键 关联 到 EventEvaluation 表 。 
ResponsibleDepartment、ResponsibleDepartment2、ResponsibleDepartment3: 第 1、 
第 2 和 第 3 责任 单位 ，nchar(2) 类 型 ， 外 键 关 联 到 Department 表 。 
ResponsiblePerson: 责任 人 ，nvarchar(10) 类 型 。 

FinishDate: 事件 完成 时 间 ，datetime 类 型 。 

ApproveLeader: 签 批 领导 ，mnvarchar(10) 类 型 。 


数据 库 中 其 他 各 表 结 构 如 图 12.1 所 示 。 


UserRole RoleRight EventType ApplicationModule 
二 RolelD 器 加 
FF ModueiD Name | Name 
Do Thepight UR 
Description 
Tspubic 
i Department EventStatus : 
User 一 到 oz 应 忻 
oD [| cepartmentName [| Name 
| Password 
UserName 
RolelD EventSource EventEvaluation 
加 加 


图 12.1 数据 库 结构 
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12.1.3 ”搭建 项 目 框架 


本 项 目 采 用 多 层 结构 设计 ， 使 用 实体 框架 完成 数据 访问 。 解 决 方案 中 应 该 包含 以 下 几 
个 项 目 : 实体 框架 层 、 数 据 访 问 层 、 业 务 罗 辑 层 、Web 表现 层 、 公 共 类 库 、 单 元 测试 项 目 、 
数据 库 管 理工 具 。 搭 建 整个 解决 方案 框架 的 步骤 如 下 : 

(1) 创建 一 个 空白 解决 方案 ， 由 于 解决 方案 中 项 目 较 多 ， 为 了 使 项 目 结构 更 加 清晰 ， 
在 解决 方案 中 添加 两 个 解决 方案 文件 夹 : DLL 和 Test， 用 以 组 织 各 个 项 目 。 

(2) 在 解决 方案 DLL 文件 夹 中 ， 添 加 一 个 类 库 项 目 SIL.Entity 作为 实体 框架 层 。 

(3) 在 解决 方案 DLL 文件 夹 中 ， 添 加 一 个 类 库 项 目 SIL.Dal 作为 数据 访问 层 。 

(4) 在 解决 方案 DLL 文件 夹 中 ， 添 加 一 个 类 库 项 目 SIL.Bll 作为 业务 逻辑 层 。 

(5) 在 解决 方案 DLL 文件 夹 中 ,添加 一 个 类 库 项 目 SIL.Common， 此 项 目 中 包含 各 个 
项 目 中 用 到 的 公共 类 。 

(6) 在 解决 方案 中 添加 一 个 ASPNET Web 应 用 程序 项 目 SIL.Web， 作 为 表现 层 。 

(7) 在 解决 方案 Test 文件 夹 中 添加 一 个 类 库 项 目 DalTest， 作 为 数据 访问 层 的 单元 测 
试 项 目 。 

(8) 在 解决 方案 Test 文件 夹 中 添加 一 个 类 库 项 目 BllTest， 作 为 业务 逻辑 层 的 单元 测试 
项 目 。 

(9) 在 各 个 项 目 之 间 添 加 引用 关系 ， 表 现 层 、 业 务 逻 辑 层 、 数 据 访 问 层 都 引用 实体 框 
架 层 和 公共 类 库 ， 业 务 罗 辑 层 引用 数据 访问 层 ， 表 现 层 引用 业务 逻辑 层 ， 测 试 项 目 引 用 被 
测试 项 目 。 整 个 解决 方案 结构 如 图 12.2 所 示 。 


Web 表 现 层 


SIL.Web 


业务 逻辑 层 BLL 测试 项 目 
SJL.BII BillTest 

数据 访问 层 DAL 测试 项 目 
SIL.Dal Dal Test 


通用 类 库 
SJL.Common 


实体 框架 层 
SJL.Entity 


上 


图 12.2 解决 方案 结构 
实体 框架 层 SIL.Entity 的 实体 框架 结构 如 图 12.3 所 示 。 
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天 tmlicatiamse- A 


日 Properties 
名 了 了 I 
Description 
Navigation Proper… Setting 
国 EventInfo © Navigation Proper… 


日 Froperties 
后 rentID 
村 randleDate 
日 properties 村 zventsource 


四 Navigation Proper' 


tt 日 Navigation Proper… 


国 RoleRieht 


省 EventType [人 A] 
EB Navigation Proper… CT 
” 国 Departnent 铭 
园 Departnentl 0..1 


了 rme 
a 日 Navigation Proper' 


国 EventInfo 


fol 
EventInfo2 
Vser 


EventSourcel 
国 EventStatus 
国 EventType 


图 12.3 实体 框架 结构 


12.2 ”主题 和 母 版 页 


一 个 正规 的 Web 应 用 程序 应 该 具有 整齐 、 规 范 、 一 致 的 界面 风格 。ASPNET 的 主题 和 
母 版 页 技术 为 维护 网 站 外 观 提 供 了 有 效 的 手段 ， 合 理 地 应 用 主题 和 母 版 ， 能 够 用 较 少 量 的 
代码 设计 出 美观 的 页 面 。 本 节 将 介绍 县 长 公开 电话 受理 系统 项 目 中 的 主题 和 母 版 页 设计 。 


12.2.1 主题 设计 


本 项 目 外 观 设计 风格 采用 浅 蓝 色 为 主 色调 ， 配 合 使 用 浅 灰 、 浅 黄 等 颜色 ， 不 使 用 非常 
鲜艳 的 颜色 ， 避 免 出 现 强烈 的 视觉 反差 ， 使 设计 出 的 页 面 干净 清新 ， 稳 重 而 不 有 呆板， 美观 
而 不 花哨 。 本 项 目 对 最 经 常 使 用 的 控件 包括 GridView、Button、TextBox、DropDownList， 
设计 了 默认 主题 和 外 观 ， 具 体 实现 步 又 如 下 : 

(1) 在 表现 层 项 目 S 和 L.Web 中 添加 一 个 名 称 为 default 的 主题 文件 夹 。 

(2) 在 default 主题 文件 夹 中 添加 一 个 外 观 文件 CommonControl.skin。 

(3) 在 CommonControl.skin 文件 中 编写 代码 ， 为 服务 器 端 控件 GridView 和 Button 指 
定 外 观 ， 其 中 Button 控件 应 用 了 jQuery UI 中 Button 的 样式 。 
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全 提示 : jQuery UI 用 到 的 CSS 文件 可 以 从 http://jqueryui.com/download 下 载 。 


<asp:GridView runat="server" CellPadding="4" ForeColor="#333333" 
GridLines="None"> 
<RowStyle CssClass="oddRow" /> 
<FooterStyle BackColor="#507CD1" Font-Bold="True" ForeColor="White" 
/> 
<PagerStyle BackColor="#2461BF" ForeColor="White" HorizontalAlign= 
"Center" /> 
<SelectedRowStyle BackColor="#D1DDF1" Font-Bold="True" ForeColor= 
"#333333" /> 
<HeaderStyle BackColor="#507CD1" Font-Bold="True" ForeColor="White" 
We 
<EditRowStyle BackColor="#2461BF" /> 
<AlternatingRowStyle CssClass="evenRow" /> 
<EmptyDataTemplate> <div style="color:red; font-size:14px; border: 
solid 1px blue; padding:30px; "> 没有 符合 条 件 的 数据 ， 请 添加 数据 或 修改 查询 
条 件 。</div> </EmptyDataTemplate> 
</asp:GridView> 
<asp:Button runat="server" CssClass="ui-button ui-state-default ui-corner-— 
all"></asp:Button> 


(4) 在 default 文件 夹 里 添加 一 个 CSS 文件 ， 在 其 中 定义 主题 中 用 到 的 样式 和 常用 
HTML 控件 样式 。 


.oddRow{ background-color:#EFF3FB;} /*GridView 奇数 行 背 景色 */ 
.evenRow{background-color:White;} /*GridView 偶数 行 背 景色 */ 
.hoverRow{background-color:#ffffdd;} /*GridView 鼠标 划 过 背景 色 */ 
input [type=text]{ border:solid lpx #9af;} /* 文 本 框 样式 */ 


input [type=text] [readonly] {background-color:#eee; } /* 只 读 文件 框 样式 */ 
select{border:solid lpx; background-color:#ffd; width:150px;} 


/* 下 拉 表 样式 */ 
上 述 主题 样式 的 外 观 如 图 12.4 所 示 。 


ELT TE 查看 Q)。 历史 名) 书签 四 ) 工具 四 帮助 A 


< pa 
Googke 耻 二 -+ 加 M: 例 全 -和 -村 :5 %-@- 
[Eee 


aaup:yAaeeahosTestroree ep 


主题 样式 预览 


fridyiew 样 式 Futton 样 式 下 拉 列表 样式 文本 样式 


i ne oe xsn | 


编 0 测试 0 其 他 数据 
编号 1 测试 1 其 他 数据 
编号 2 测试 2 其 他 数据 
编号 3 测试 3 其 他 数据 
编号 4 测试 4 其 他 数据 


[a Eat 


| 
如 


图 12.4 主题 样式 预览 


12.2.2 ” 母 版 页 设计 
公开 电话 受理 系统 中 页 面 布局 分 为 3 部 分 : 页 头 、 正 文 和 页 脚 ， 其 中 页 头 显示 工具 栏 ， 
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页 脚 显示 网 站 说 明和 联系 方式 ， 这 两 部 分 内 容 对 于 大 多 数 页 面 来 说 是 相同 的 ， 属 于 母 版 页 
的 内 容 。 公 开 电 话 受理 系统 母 版 页 的 实现 步骤 如 下 : 
(1) 在 表现 层 项 目 SIL.Web 中 添加 一 个 文件 夹 UserControls， 用 于 存放 项 目 中 的 用 户 
控件 。 
(2) 在 UserControls 文件 夹 中 添加 一 个 用 户 控件 Headerascx， 用 于 实现 页 面 头 部 ， 其 
中 主要 包含 简单 的 提示 信息 和 功能 导航 栏 。Headerascx 用 户 控件 设计 外 观 如 图 12.5 所 示 。 


_A_ 有 具 书 记 县 长 公开 电话 受理 平台 当 症 日 期 ， 当 前 用 户 ， 【退出 登录 】 


忠 


图 12.5 Header 用 户 控件 


Headerascx 用 户 控件 代码 如 下 : 


<%-- 这 个 div 显示 程序 名 称 、 当 前 日 期 和 当前 用 户 --$> 
<div style="background:#f2f2fe; width:1000px; text-align:left;"> 
<div> 
<span style="color:#005599; font-size:24px; font-weight:bold; font-family: 
华文 楷体 ;”> 
A 县 书记 、 县 长 公开 电话 受理 平台 </ span> 
<span style="margin-left:120px; margin-right:30px;"> 


当前 日 期 : 

<%$=DateTime .Now.ToLongDateString ()+"&nbspz"+DateTime.Today.DayOfWeek $%> 
当前 用 户 : 

<%= SJL.Web.HttpCode.WebUtility.currentUser == null ? "未 知 用 户 " 


SuL.Web.HttpCode .WebUtility.currentUser.UserNames> 

&nbsp; &nbsp; 【<a href=". ./LogoutPage .aspx"> 退 出 登录 </a>】 </span> 

</div> 

</div> 

<&%-- 下 面 这 个 div 为 功能 导航 栏 --%> 

<div id="NavMenu"> 

<div class="ImageMenu"><img src="../images/policy.png" alt=" 电 话 业 务 受 理 " 
/><br/> 

<a href=". ./Default/EventEditPage .aspx?id=-1"> 电 话 业 务 受理 </a></div> 

<div class="ImageMenu"><img src="../images/report.png"” alt=" 业 务 详 情 查询 " 
/><br/> 

<a href="../Default/EventListPage.aspx"> 受 理 业务 查询 </a></div> 

<div class="ImageMenu"><img src="../images/user.png”alt=" 用 户 权 限 管理 " 
/><br/> 

<a href="../UserRight/ChangePassPage.aspx"> 用 户 权限 管理 </a></div> 

<div class="ImageMenu"><img src="../images/dictionary.png"”alt=" 数 据 字 典 管 
DE/ 

<a href=". ./Dictionary/DepartmentPage .aspx"> 数 据 字 典 管理 </a></div> 

</div> 


在 Headerascx 用 户 控 件 中 ， 用 CSS 样式 控制 导航 菜单 向 左 浮动 以 及 文字 和 图 片 的 上 
下 对 齐 方式 ，CSS 代码 如 下 : 
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/*div#NavMenu 页 面 顶部 导航 菜单 */ 

div#NavMenu{background-color: #003399; font-size:medium; 
font-weight:bold; height:70px; vertical-align:middle;} 

div#NavMenu a {color:White; } 

div#NavMenu a:hover{ background-color:#ddddff; color:black; } 

/* 带 图 片 的 菜单 ， 上 图 片 ， 下 文字 */ 

div#NavMenu div.ImageMenu{ text-align:center; float:left; margin:auto 
TOP 

div#NavMenu div.ImageMenu img{height:48px;} 


(3) 在 UserControls 文件 夹 中 添加 一 个 用 户 控 件 Footerascx， 用 于 实现 网 站 页 脚 。 


<%@ Control Language="C#" AutoEventWireup="true" CodeBehind="Footer.ascx. 
cs" Inherits="SJL.Web.UserControls.Footer" %> 
<div style="width:1000px; text-align:center; 
border:dashed lpx silver; background:#f2f2f2; float:none; clear:both;"> 
A 县 书记 、 县 长 公开 电话 受理 平台 。 公 开 电 话 : (0543) 5350686; 5326899。<br /> 
主办 单位 ;滨州 市 A 县 人 民政 府 。 技 术 支 持 : <a href= 
"mailto:sun.j.1.studio@gmail.com"> sun.j.1.studio@gmail.com </a> 
</div> 


(4) 在 SIL.Web 项 目 中 添加 一 个 母 版 页 SjIMaster.Master， 在 母 版 页 中 放置 3 个 div， 
在 第 1 个 div 中 放置 Header 用 户 控 件 , 第 2 个 div 中 放置 一 个 ContentPlaceHolder, 第 3 个 
div 中 放置 Footer 用 户 控件 ， 代 码 如 下 : 

<div id="main"> 

<form id="forml" runat="server"> 

<div> <ucl:Header ID="Headerl" runat="server" /> </div> 

<div> 

<asp:ContentPlaceHolder ID="content" runat="server"> </asp:ContentPlace- 

Holder> 

</div> 

<div> <uc2:Footer ID="Footerl" runat="server" /> </div> 

</form> </div> 


为 了 使 整个 页 面 在 不 同 分 辨 率 下 保持 固定 大 小 ， 并 且 在 浏览 器 中 居中 显示 ， 需 要 对 页 
面 最 外 层 的 div 进行 样式 设置 ， 设 定 其 宽度 、 水 平 对 齐 方式 ，CSS 代码 如 下 : 

/* 最 外 层 DIV， 整 个 页 面 居 中 */ 

divi#main{width:1000px; margin-left:auto; margin-right:auto; } 

(5) 在 母 版 页 SjIMaster.master 中 创建 一 个 内 容 页 面 并 测试 母 版 页 外 观 ， 运行 界面 如 图 
12.6 所 示 。 


Ssturday 当前 用 户 。 管理 只。 【过 出 登录 】 


欢迎 使用 公开 电话 受理 平台 

请 从 页 耐克 节 音 栏 中 迁 拉 所 于 进行 特 接 作 
县 县 书记 生长 公 开 电 话 受 理 平 台 。 公 开 电 话 : (010) 12145678: B1654321。 
主 坟 单 位 ，_E 市 -上 县 人 民政 府 。 技 术 支持 。 sun j.1. =tudiogemail.con 


E23 Eal: : 


图 12.6 母 版 页 外 观 
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12.3 ”电话 业务 受理 


当 有 群众 通过 县 长 公开 电话 反映 问题 时 ， 公 开 电话 受理 中 心 的 工作 人 员 需 要 将 问题 的 
详情 通过 程序 记录 下 来 。 本 节 将 讨论 这 一 功能 的 设计 和 实现 。 


12.3.1 事件 编号 生成 算法 


在 县 长 公开 电话 受理 系统 项 目 中 ， 和 群众 通过 电话 所 反映 的 一 个 问题 称 为 一 个 事件 ， 寻 
件 信 息 保存 在 数据 库 的 EventInfo 表 中 。 在 添加 新 事件 时 ， 需 要 为 新 事件 生成 一 个 唯一 的 
件 编号 ， 从 数据 库 的 角度 来 说 ， 要 为 这 条 新 记录 确定 一 个 主键 。 生 成 事件 编号 最 简单 的 办 
法 就 是 使 用 数据 库 的 标识 列 〈 自 动 增长 列 ), 但 是 这 样 生成 的 主键 可 读 性 很 差 。 在 实际 应 用 
中 ， 通 用 使 用 具有 一 定 字面 含义 的 文本 作为 主键 ， 例 如 ， 可 能 用 某 种 算法 基于 当前 日 期 生 
成 一 个 字符 串 作 为 主键 。 

在 公开 电话 受理 项 目 中 ， 使 用 当前 日 期 (yymmdd 格式 的 6 位 数字 ) 加 4 位 顺序 号 组 
合 为 10 位 数字 作为 主键 ， 例 如 0910030212、1002050018 等 。 每 当 添 加 一 个 新 的 事件 信息 
时 ， 都 需要 根据 此 算法 为 新 事件 生成 一 个 主键 。 为 了 使 顺序 号 递增 ， 需 要 记录 上 一 个 事件 
的 顺序 号 , 将 上 一 事件 顺序 号 加 1 即 可 得 到 当前 事件 顺序 号 , 当 顺 序号 增加 到 9999 时 恢复 
为 0。 为 了 提高 查询 速度 和 简化 顺序 号 生成 算法 , 在 数据 库 中 使 用 一 个 表 ApplicationSetting 
来 保存 当前 顺序 号 。 根 据 以 上 分 析 ， 新 增 事件 编号 生成 算法 如 下 : 

(1) 取得 当前 日 期 ， 生 成 yymmdd 格式 的 6 位 数字 。 

(2) 从 数据 库 ApplicationSetting 表 中 取得 当前 顺序 号 ， 转 换 为 4 位 数字 ， 不 足 4 位 数 
字 的 左 侧 用 0 补 齐 。 

(3) 将 数据 库 ApplicationSetting 表 中 当前 顺序 号 加 1 并 保存 ,如 果 顺 序号 加 1 超过 9999 
(4 位 整数 能 表示 的 最 大 值 )， 则 将 顺序 号 重 置 为 0 并 保存 到 数据 库 。 

(4) 将 6 位 日 期 与 4 位 顺序 号 组 合 ， 即 得 到 新 事件 编号 。 

事件 编号 生成 算法 对 应 代码 如 下 (位 于 数据 访问 层 项 目的 EventmfoDAL 类 中 ): 

public static string getNewEventID () 

{ 
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var context = new UserRightContext();  // 创 建 实体 容器 

int n=context.getPartialNewEventID();  // 调 用 存储 过 程 得 到 事件 顺序 号 

string s=DateTime.Now.ToString ("yyMMdd") + n.ToSstring("0000"); 
// 得 到 完整 事件 编号 

context.Dispose(); 

return ss 


| 
上 述 代 码 中 用 到 的 存储 过 程 sp_GetPatialINewEventID 代码 如 下 : 


CREATE PROCEDURE [dbo].[sp GetPartialNewEventID] 
AS 
BEGIN 

set nocount on 

declare ret int 


-3 


begin transaction 
select ret=cast (Setting as int) from ApplicationSetting where 
ID=17 
if @ret>=9999 set ret=1; 
update ApplicationSetting set Setting=@ret+l] where ID=1; 
commit transaction 
select eret 
END 


12.3.2 ”数据 访问 层 和 业务 逻辑 层 


公开 电话 受理 系统 是 一 个 多 层 结构 的 Web 应 用 程序 , 与 数据 库 相 关 的 代码 都 在 数据 访 
问 层 中 实现 。 为 了 实现 电话 业务 受理 功能 , 需要 在 数据 访问 层 中 实现 相应 的 数据 操作 功能 ， 
主要 包括 事件 的 添加 、 修 改 和 查询 。 

(1) 在 数据 访问 层 项 目 S 了 L.Dal 中 添加 一 个 类 EventInfoDal。 


public static class EventInfoDAL 


(2) 为 提高 数据 访问 层 代 码 的 可 靠 性 ， 尽 早 发 现 程 序 中 的 bug， 应 该 在 第 1 时 间 对 数 
据 访 问 层 代码 进行 单元 测试 。 在 测试 项 目 DalTest 中 添加 一 个 新 的 单元 测试 类 
EventInfoDalTest， 此 类 将 测试 EventInfoDal 中 的 方法 。 


[TestClass] 
public class EventInfoDalTest 
oe 


(3) 在 EventInfoDal 类 中 添加 一 个 getByID0 方 法 ， 实 现 根据 事件 ID 取得 事件 详情 的 
功能 。 


public static EventInfo getByID(string id) // 根 据 事件 ID 得 到 事件 信息 
{ 

Var query = getQuery(); 

var target= EntityUtility.selectOne (query,e=>e.EventID==id, false, 

false); 

loadReference (target); 

EntityUtility.detachEntity (query.Context, target); 

query.Context .Dispose(); 

return target; 


} 
(4) 在 EventInfoDalTest 类 中 添加 一 个 方法 , 对 EventInfoDAL.getByID0 方 法 进行 测试 。 


[TestMethod] 
public void testGetByID() // 根 据 事件 ID 得 到 事件 信息 
| 
Enimo dr OO0lem // 一 个 已 经 存在 的 事件 ID 
Var e = EventInfoDAL.getByID(id); 
Assert.IsNotNull (e); // 所 返回 的 事件 不 为 空 
Assert.IsNotNull (e.HandleUser); // 事 件 受理 者 不 为 空 
Rssert.RAreEqual (id，e-EventID) // 事 件 ID 正确 


} 
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进行 下 一 步 。 
(6) 在 EventInfoDal 中 添加 一 个 add0 方 法 ， 实 现 添 加 事件 的 功能 。 
// 添 加 事件 


public static int add(EventInfo info) 


return EntityUtility.add<UserRightContext,EventInfo> (info); 
} 


(7) 在 EventInfoDalTest 中 添加 一 个 方法 ， 对 EventInfoDAL.add0 方 法 进行 测试 。 


[TestMethod] 
public void testadd () 


// 创 建 一 个 新 事件 并 设置 其 主要 属性 
EventInfo e = new EventInfo() 
string id=EventInfoDAL .getNewEventID() 
e-EventID = id; 
e.EventTitle = "test"; 
e.HandleUser = "admin"; 
e.HandleDate = DateTime.Now; 
EventInfoDAL.add (e); // 添 加 事件 
var e2 = EventInfoDAL.getByID(id); // 获 取 添 加 的 事件 
Assert .IsNotNull (e2); 
} 


(8) 运行 EventInfoDALTest.testAdd 测试 方法 ,如 有 错误 则 及 时 修改 , 测试 通过 后 进行 
(9) 在 EventInfoDal 类 中 添加 一 个 update0 方 法 ， 实 现 修改 事件 的 功能 。 


// 修 改 事件 

public static int update (EventInfo info) 

{ 
return EntityUtility.update<UserRightContext, EventInfo> (info, "Event 
ED 
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(10) 在 EventInfoDalTest 类 中 添加 一 个 方法 , 对 EventInfoDAL.update0 方 法 进行 测试 。 


[TestMethod] 
// 修 改 事件 
public void testUpdate () 
1 
string id = "0018"; // 一 个 已 经 存在 的 事件 ID 
var e = EventInfoDAL.getByID (id); // 首 先 取得 事件 
// 以 下 语句 修改 事件 各 个 属性 
string department = "02"; 
int evaluation = 3; 
string person = "zx02"; 
e.EvaluationID = evaluation; 
e.ResponsibleDepartment = department; 
e.ResponsiblePerson = person; 
EventInfoDAL.update (e); // 将 事件 更 新 到 数据 库 
e = EventInfoDAL.getByID (id); // 重 新 从 数据 库 取得 事件 
// 以 下 语句 测试 所 取得 的 事件 信息 已 经 变 成 修改 后 的 值 
Assert.AreEqual (department, e.ResponsibleDepartment); 
Assert.AreEqual (person, e.ResponsiblePerson); 
Assert.AreEqual (evaluation, e.EvaluationID); 
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(11) 在 EventInfoDal 类 中 添加 一 个 delete0 方 法 ， 实 现 删除 事件 功能 。 
// 删 除 事件 


public static int delete(string id) 


{ 
} 


return EntityUtility.delete (getQuery (),e=>e.EventID==id); 


(12) 在 EventInfoDalTest 类 中 添加 一 个 方法 ， 对 EventInfoDAL .delete0 方 法 进行 测试 。 


[TestMethod] 
public void testDelete () 


/ /为 避免 删除 数据 库 中 原 有 数据 ， 先 添加 一 个 新 事件 然后 再 删除 
EventInfo e = new EventInfo() :7 

string id = EventInfoDAL.getNewEventID(); 
e.EventID = id; 

e.EventTitle = "test"; 


e.HandleUser = "admin"; 

e.HandleDate = DateTime.Now; 

EventInfoDAL.add (e); // 添 加 事件 

Var e2 = EventInfoDAL.getByID (id); // 获 取 添 加 的 事件 


Assert .IsNotNull (e2); 

EventInfoDAL.delete (id); 

e2 = EventInfoDAL.getByID (id); // 执 行 删除 

Assert .IsNull (e2); /7 删除 后 数据 库 中 不 存在 此 事件 


(13) 业务 轴 辑 层 的 功能 简单 ， 只 需要 调用 相应 的 数据 访问 层 方法 即 可 。 在 业务 逻辑 
层 项 目 SL.B1 中 添加 一 个 类 EventmfoBLL， 在 类 中 添加 增 、 删 、 改 、 查 功能 。 


public static class EventInfoBLL 


{ 


“ 308" 


private const string NewEventID = ""; 
[DataObjectMethod (DataObjectMethodType.Insert)] 
public static int add(EventInfo info) 
{ 
if (string.IsNullOrEmpty (info.EventID)) 
throw new ArgumentNullException ("EventID") 
return DB.add (info); 
} 
public static int update (EventInfo info) 
1 
return DB.update (info); 
} 
public static int delete (string id) 
{ 
return DB.delete(id); 
} 
public static EventInfo getByID(string id) 
{ 
return DB.getByID (id); 
} 
public static EventInfo newEventInfowWithID() 
1 
return new EventIinfo() { EventID = NewEventID}; 


} 
public static string getNewEventID() 


第 12 章 县 长 公开 电话 受理 系统 


return DB.getNewEventID(); 


12.3.3 ”事件 详情 用 户 控件 


在 公开 电话 受理 系统 项 目 中 ， 会 在 多 个 页 面 中 显示 、 修 改 、 添 加 事件 信息 。 为 了 避免 
重复 代码 ， 遵 循 一 个 功能 只 实现 一 次 的 原则 ， 应 该 将 事件 详情 显示 和 编辑 界面 封装 成 一 个 
用 户 控 件 ， 再 放 到 不 同 的 页 面 中 使 用 。 创 建 事件 详情 用 户 控件 步骤 如 下 : 

(1) 在 表现 层 项 目 SL.Web 中 添加 一 个 文件 夹 UserControls， 用 来 存放 项 目 中 的 所 有 
Web 用 户 控 件 。 在 UserControls 文件 夹 事 件 中 新 建 一 个 用 户 控件 EventInfoControl.ascx， 
参照 图 12.7 对 页 面 进行 设计 。 


NiddenField - hidden0ldTD 


事项 编号 受理 日 期 受理 类 别 [Unbound 到 受理 人 员 
反映 人 电话 地 址 清 意 程度 Unbound 司 
承办 单位 [Unbound 。 ， 司 。 承 办 人 员 承办 单位 2 [Unbound 司 承办 单位 3 [Unbound 可 
问题 类 型 [Unbound 到 ”处 理 状态 [Unbound 司 办 结晶 期 签 批 领导 
反映 主题 条] ] 

-本 

| 

反映 内 容 | 


图 12.7 事件 详情 用 户 控 件 


在 图 12.7 所 示 的 事件 详情 用 户 控 件 EventInfoControl.ascx 中 , 整个 页 面 使 用 table 布局 ， 
受理 日 期 和 办 结 日 期 都 启用 了 jQuery 日 历 控件 ， 反 映 内 容 和 答复 意见 分 别 是 一 个 多 行 的 
TextBox 控件 ， 用 一 个 隐藏 域 HiddenField 保存 当前 正在 编辑 的 事件 编号 。 
EventInfoControl.ascx 代码 如 下 : 


<%@ Control Language="C#" AutoEventWireup="true" CodeBehind="EventInfo 
Control.ascx.cs" Inherits="SJL.Web.UserControls.EventInfoControl" $%> 
<script type="text/javascript"> 
$ (function() { 
$('#<%=theDate.ClientID%$>') .datepicker(); // 启 用 jQuery 日历 控 件 
$('#<%=finishDate.ClientID$>') .datepicker (); 
1D); 
</script> 
<table style="background:#eff; "> 
<Er> 
<td colspan="8"> 
<asp:HiddenField ID="hiddenOldID" runat="server" /> 
</td> 
</tr> 
<tr><td> 事 项 编号 </td> 
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<td><asp:TextBox runat="server" ID="eventId" Text=""></asp:TextBox></td> 
<td> 受 理 日 期 </td> 

<td><asp:TextBox ID="theDate" runat="server" Text="" ></asp:TextBox></td> 
<td> 受 理 类 别 </td> 

<td> <asp:DropDownList ID="eventSourceList" runat="server"> </asp:Drop- 
DownList></td> 

<tq> 受 理 人 员 </tq> 

<td> <asp:TextBox ID="handleUser" runat="server"></asp:TextBox></td></tr> 
<tr><tqd > 反映 人 </td> 

<td> <asp:TextBox ID="personName" runat="server"></asp:TextBox> </td> 
<td> 电 话 </td> 

<td> <asp:TextBox ID="personPhone" runat="server"></asp:TextBox></td> 
<td> 地 址 </td><td> <asp:TextBox ID="personAddress" runat="server"></asp: 
TextBox> </td> 

<td> 满 意 程度 </td><td><asp:DropDownList runat="server" ID="evaluationList" 
/></td></tr> 

<tr><td> 承 办 单位 </td> 

<td><asp:DropDownList ID="departmentList" runat="server" DataValueField= 
"DepartmentID" DataTextField="DepartmentName"> </asp:DropDownList></td> 
<td> 承 办 人 员 </td> 

<td> <asp:TextBox ID="responsiblePerson" runat="server"></asp:TextBox> 
</td> 

<tq> 承 办 单位 2</td> 

<td><asp:DropDownList ID="departmentList2" runat="server" DataValueField= 
"DepartmentID" DataTextField="DepartmentName"> </asp:DropDownList></td> 
<td> 承 办 单位 3</td> 

<td><asp:DropDownList ID="departmentList3" runat="server" DataValueField= 
"DepartmentID" DataTextField="DepartmentName"> </asp:DropDownList></td> 
</tr> 

<tr> <tqd> 问 题 类 型 </td> 

<td><asp:DropDownList ID="eventTypeList" runat="server"></asp:DropDown- 
List></td> 

<td> 处 理 状态 </td> 

<td><asp:DropDownList ID="eventStatusList" runat="server"> </asp:DropDown 
List></td> 

<td> 办 结 日 期 </td> 

<td> <asp:TextBox ID="finishDate" runat="server" Text="" ></asp:TextBox> 
</td> 

<td> 签 批 领导 </td> 

<td> <asp:TextBox ID="leader" runat="server"></asp:TextBox> </td> </tr> 
<tr><td> 反 映 主题 </td> 

"><asp:TextBox ID="eventTitle" runat="server" Width="800"> 
</asp: TextBox></td></tr> 

<tr><td> 反 映 内 容 </td> 

<td colspan="7"> <asp:TextBox ID="eventContent" runat="server" Height= 
"200px" Width="800px" 

TextMode="MultiLine"> </asp:TextBox> </td> </tr> 

<tr><td> 答 复 意 见 </td> 

<td colspan="7"> <asp:TextBox ID="eventResult" runat="server" 
Height="120px" Width="800px" 

TextMode="MultiLine"></asp:TextBox> </td> </tr> 

</table> 


(2) 在 用 户 控件 EventInfoControl.ascx 中 有 多 个 下 拉 列 表 DropDownList 控件 ， 包 括 部 
门 列表 、 问 题 类 型 列表 、 处 理 状态 列表 、 满 意 程度 列表 。 这 些 下 拉 列 表 控 件 都 需要 绑 定 到 
数据 库 中 的 数据 ， 一 般 用 如 下 代码 实现 〈 以 问题 类 型 列表 为 例 ): 


“0 
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则 可 


var list=EventTypeBl]l.getAll(); // 从 数据 库 中 得 到 问题 类 型 数据 
eventTypeList.DataValueField="ID"; // 设 置 值 字段 
eventTypeList.DataTextField="Name"; // 设 置 显示 字段 


eventTypeList.DataSource=list; 
eventTypeList.DataBind(); 


用 户 控 件 EventInfoControl.ascx 中 有 4 个 下 拉 表 ， 如 果 为 每 个 下 拉 表 都 写 同样 代码 ， 


E 复 代码 太 多 ， 而 且 其 他 页 面 中 也 存在 类 似 的 很 多 下 拉 表 。 利 用 泛 型 和 接口 可 以 编写 一 


个 通用 方法 实现 数据 绑 定 功能 ， 如 下 代码 所 示 : 


// 假 设 数据 字典 只 有 两 个 属性 〈 字 段 ) : ID 和 Name 
// 数 据 字 典 业务 罗 辑 层 只 有 一 个 方法 : 返回 所 有 数据 
public interface IDictionaryB11<T> // 数 据 字 典 业 务 罗 辑 层 接口 
{ 
List<T> getAll (); 
. 
public class EventEvaluationB11:IDictionaryB11<EventEvaluation> 
// 事 件 评价 业务 逻辑 层 
{ 
public List<EventEvaluation> getAll() 
i 
return EventEvaluationDal .getAll (); 
} 
} 
public class EventSourceB11:IDictionaryB11<EventSource> 
// 事 件 来 源 业 务 罗 辑 层 
public List<EventSource> getAll() 
{ 
return EventSourceDal .getAll(); 
| 
Y 
public class EventStatusBll:IDictionaryBll<EventStatus> 
// 事 件 处 理 状态 业务 逻辑 层 
{ 
public List<EventStatus> getRl1l() 
{ 
return EventStatusDal .getAll(); 
} 
} 
public class EventTYpeB11:IDictionaryB11<EventType>// 事 件 类 型 业务 罗 辑 层 
public List<EventType> getAll() 
{ 
return EventTypeDal .getAll(); 
} 
; 
//Web 工具 类 ， 包 含 几 个 常用 方法 
public static class WebUtility 
///<summary> 
/// 将 数据 字典 绑 定 到 下 拉 列 表 控件 中 
///</summary> 
///<typeparam name="TB11"> 获 取 数据 字典 数据 的 业务 逻辑 层 类 型 </typeparam> 
///<typeparam name="TElement"> 业 务实 体 类 型 </typeparam> 
///<param name="dropdown"> 要 绑 定 的 下 拉 列 表 控件 </param> 
public static void bindListWithDictionary<TB]11, TElement> (DropDownList 
a 
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mn 


where TB11 : IDictionaryBll<TElement>, new() 


var 1= new TB11() .getAll (); // 调 用 业务 逻辑 层 方法 返回 数据 
list.DataValueField = "id"; 

list.DataTextField = "name™"; 

list.DataSource = 1; 

list.DataBind(); 


} 
(3) 在 用 户 控 件 EventInfoControl.ascx 的 Page Load 事件 中 ， 利 用 上 一 步 所 实现 的 通 
上 数据 绑 定 方法 将 数据 绑 定 到 下 拉 列 表 中 : 


protected void Page Load (object sender, EventArgs e) 


if (!IsPostBack) 
initList() 7 
// 下 拉 列 表 是 否 已 经 绑 定 数据 〈 防 止 重复 绑 定 ) 
private bool listInitialized 


{ 
get { return WebUtility.convertToBoolean (ViewState [ViewStateInit- 


Vise 
set { ViewState[ViewStateInitList] = value; } 
.| 
// 为 下 拉 列 表 填 充 数据 
private void initList() 
{ 
if (listInitialized) return; 
// 下 面 4 条 语句 调用 泛 型 方法 绑 定数 据 列表 
WebUtility.bindListWithDictionary<EventSourceB11，EventSource> (event 
SourceList); 
WebUtility.bindListWithDictionary<EventTypeBll1l, EventType> (eventTyp- 
eList); 
WebUtility.bindListWithDictionary<EventStatusBll, EventStatus> (event— 
StatusList); 
WebUtility.bindListWithDictionary<EventEvaluationBl]l, EventEvaluati-— 
on> (evaluationList); 
initDepartmentList(); 
listInitialized = true; 
;》 
// 为 部 门 列表 填充 数据 
private void initDepartmentList() 
| 
if (listInitialized) return; 
var departments=DepartmentBLL.getAll (PageDataArgument .allData); 
DropDownList[] lists={departmentList,departmentList2,departmentL-— 
Lat 
for (Tn 0 < 3) // 将 部 门 列表 绑 定 到 3 个 下 拉 列 表 中 
1 
lists[i].DataSource= departments; 
lists[i] .DataBind(); 
// 在 部 门 列表 的 第 1 项 插入 "请 选择 "字样 
lists[i] .Items.Insert(0，new ListItem("-- 请 选择 --",，"-1")); 
lists[i].SelectedIndex = 0; 


(4) 用 户 控件 EventInfoControl.ascx 用 于 显示 和 编辑 事件 详情 ， 当 保存 事件 信息 时 ， 
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需要 区 分 当前 被 编辑 的 事件 是 否 为 空 ， 是 新 增 事件 ， 还 是 原 有 事件 。 可 以 在 用 户 控件 中 添 


加 属性 以 区 分 事件 数据 的 不 同 状态 : 


public partial class EventInfoControl : System.Web.UI.UserControl 
{ 

#region constants 

// 以 下 常量 皆 为 Viewstate 中 的 键 值 

private const string ViewStateIsNewRecord = "isNewRecord"; 
private const string ViewStateIsNullRecord = "isNullRecord"; 
private const string ViewStateInitList = "initlist"; 

private const string ViewStateTheEvent = "theEvent"; 
#endregion 

// 用 户 控件 当前 显示 的 是 否 一 条 新 记录 

private bool isNewRecord 


{ 


get 

{ 
return WebUtility.convertToBoolean (ViewState [ViewStateIsNewRec- 
ord]); 

} 

Set 

i 
ViewState[ViewStateIsNewRecord] = value; 


} 
» 
// 用 户 控件 当前 显示 的 是 否 一 条 空 记录 
private bool isNull 
{ 
get 
{ 
return WebUtility.convertToBoolean (ViewState [ViewStateIsNull 
ord]); 


Se€E 
{ 
ViewState[ViewStateIsNullRecord] = value; 
} 
} 
3» 


(5) 在 用 户 控 件 EventInfoControl.ascx 中 添加 一 个 EventInfo 类 型 的 属性 ， 实 现 用 


面 和 实体 类 之 间 的 双向 数据 传递 : 


// 与 此 用 户 控件 相对 应 的 EventInfo 
public EventInfo theEvent 
| 
get { return getEventIinfo(); } 
set { setEventInfo(value); } 
/// <summary> 
/// 根据 用 户 控件 生成 事件 信息 (一 个 EventInfo 类 的 实例 ) 
/// </summary> 
/// <returns> 得 到 的 事件 信息 </returns> 
private EventInfo getEventInfo() 
{ 


Rec= 


if (isNull) return null; // 如 果 为 空 记录 则 返回 nul1 


if (!1istInitialized) 
initLbist(); 


if (ViewState [ViewStateTheEvent] == null)  // 将 事件 信息 保存 在 ViewState 
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; 


ViewState [ViewStateTheEvent] = new EventInfo(); 
EventInfo info = ViewState[ViewStateTheEvent] as EventInfo; 
// 以 下 语句 根据 页 面 上 各 控件 的 值 设 置 EventInfo 各 个 属性 
info.EventID = eventId.Text; 
info.HandleDate = DateTime.Parse (theDate .Text) ; 
info.HandleUser = handleUser.Text; 
info.EventSource = int.Parse (eventSourceList.SelectedValue); 
info.PersonName = personName.Text; 
info.TypeID = int.Parse (eventTypeList.SelectedValue); 
info.EventContent = eventContent.Text; 
info.StatusID = int.Parse (eventStatusList.SelectedValue); 
string department; 
department=departmentList.SelectedValue; 
info.ResponsibleDepartment = department == "-1" ? "" : department; 
info.ResponsiblePerson = responsiblePerson.Text; 
department = departmentList2.SelectedValue; 


info.ResponsibleDepartment2 = department == "-1" ? null : department; 
department = departmentList3.SelectedValue; 
info.ResponsibleDepartment3 = department == "-1" ? null : department; 


info.EvaluationID = int.Parse (evaluationList.SelectedValue); 

info.PersonAddress = personAddress.Text; 

info.PersonPhone = personPhone.Text; 

info.EventResult = eventResult.Text; 

info.EventTitle = eventTitle.Text; 

info.ApproveLeader = leader.Text; 

DateTime t; 

if (DateTime.TryParse (finishDate.Text，out t)) // 判 断 是 否 输入 合法 日 期 
info.FinishDate = t; 

else 
info.FinishDate = null; 

return info; 


/// <summary> 

/// 根据 事件 信息 设置 用 户 控件 

/// </summary> 

/// <param name="info"> 要 设置 的 事件 信息 </param> 


private void setEventInfo (EventInfo info) 


{ 


.394 。 


if (!listInitialized) 
initList(); 
isNewRecord = false; 


ViewState [ViewStateTheEvent] = info; 
if (info = null) // 如 果 事 件 为 空 则 清空 界面 
{ 
isNull = true; 
hiddenoldID.Value = ""; 
WebUtility.clearFormInput (this); 
return; 
} 
hiddenoldID.Value = info.EventID; // 保 存 原 事件 ID 


isNull = false; 

// 以 下 语句 根据 EventInfo 的 各 个 属性 设置 相应 控件 内 容 

eventId.Text = info.EventID; 

theDate.Text = info.HandleDate.Value.ToShortDatestring(); 
handleUser.Text = info.HandleUser; 

// 根 据 指定 的 值 选中 下 拉 列 表 中 某 一 项 

WebUtility.selectByValue (eventSourceList,info.EventSource .To-- 
String()); 

personName.Text = info.PersonName; 
WebUtility.selectByValue (eventTypeList, info.TypeID.ToString()); 
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eventContent -Text = info.EventContent; 

WebUtility.selectByValue (eventstatusList, info.StatusID.ToString()); 
WebUtility.selectByValue (departmentList, info.ResponsibleDepartment); 
WebUtility.selectByValue (departmentList2, info.ResponsibleDepartm-— 


ent2); 

WebUtility.selectByValue (departmentList3, info.ResponsibleDepartm-— 
ent3); 

WebUtility.selectByValue (evaluationList, info.EvaluationID.ToStr-— 
ing()); 


responsiblePerson.Text = info.ResponsiblePerson; 
personPhone.Text = info.PersonPhone; 
personAddress.Text = info.PersonAddress; 
eventResult.Text = info.EventResult; 
eventTitle.Text = info.EventTitle; 
if (info.FinishDate.HasValue) 
finishDate.Text = info.FinishDate.Value.ToShortDatestring(); 
else 
finishDate.Text = ""; 
leader.Text = info.ApproveLeader; 


1 


上 述 代 码 用 到 了 一 个 WebUtility 类 中 的 一 个 静态 方法 selectByValue()， 这 个 方法 的 作 
用 是 根据 指定 的 值 从 下 拉 列 表 控 件 中 选中 匹配 项 ， 如 下 代码 所 示 : 


/// <summary> 

/// 从 下 拉 列 表 中 选中 具有 指定 值 的 一 项 

/// </summary> 

/// <param name="1ist"> 下 拉 列 表 控件 </param> 

/// <param name="value"> 要 选 定 的 值 </param> 

/// <returns> 所 选中 的 项 的 索引 ， 如 果 不 存 在 对 应 项 则 返回 -1</returns> 
public static int selectByValue (DropDownList list,string value) 


{ 


if(list.Items.Count==0) 

return = 
if (!string.IsNullOrEmpty (value)) 
{ 


for (int i = 0; i < list.Items.Count; i++) 


if (1ist.Items[i].Value == value) // 找 到 指定 值 
{ 
list.SelectedIndex = i; 
return i; 
} 
1 


} 
list.SelectedIndex = 0; 
etiire 17 


(6) 在 用 户 控 件 EventInfoControl.ascx 中 添加 一 个 方法 setByIDO， 实 现 根据 事件 编号 
填充 控件 内 容 的 功能 : 


/// <summary> 
/// 根据 事件 编号 设置 用 户 控件 内 容 
/// </summary> 
/// <param name="id"> 事 件 编号 </param> 
public void setByID(string id) 
) 
if (string.IsNullOrEmpty (id)) // 如 果 编 号 为 空 则 清空 控件 


-0 


setEventInfo (null); 
else 
setEventInfo (EventInfoBLL.getByID(id)); 


// 否 则 调用 业务 逻辑 层 得 到 事件 信息 并 设置 控件 内 容 
} 


(7) 在 用 户 控 件 EventInfoControl.ascx 中 添加 一 个 save0 方 法 ， 用 以 保存 事件 数据 。 在 
保存 时 需要 判断 此 事件 是 新 增 事件 还 是 原 有 事件 ， 从 而 分 别 调用 业务 逻辑 层 的 add 或 者 
update() 方 法 实现 添加 或 更 新 。 在 更 新 原 有 数据 时 ， 还 要 判断 主键 事件 编号 是 否 发 生 改 变 ， 
如 果 发 生 了 改变 ， 则 需要 先 将 原 有 事件 删除 ， 再 添加 新 的 事件 : 

/// <summary> 

/// 保存 (插入 或 更 新 ) 事件 信息 

/// </summary> 


public int save() 
{ 


var e = theEvent; 


if (e == null) return 0; 
int n= 0; 
if (isNewRecord) // 新 添加 的 事件 


{ 
n = EventInfoBLL.add(e) > 
eventId.Text = e.EventID; 


else // 原 有 事件 
{ 
if (keyChanged) // 主 键 发 生 了 改变 
{ 
n = EventInfoBLL.add (e); // 插 入 新 记录 


n = EventInfoBLL.delete (hiddenoldID.Value) ; // 删 除 旧 记 录 


else 
n 


EventInfoBLL.update (e); 
} 
isNewRecord = false; 
return n; 
有 
/// <summary> 
/// 事件 编号 是 否 发 生 了 改变 
/// </summary> 
private bool keyChanged 
{ 
get 
{ 
if (isNewRecord) return true; 
// 通 过 比较 HiddenField 保存 的 原 有 事件 编号 与 TextBox 控件 中 的 事件 编号 确定 是 否 
// 改 变 
return hiddenOldID.Value != eventId.Text.Trim(); 
} 


12.3.4 ”电话 业务 受理 页 面 


电话 业务 受理 页 面 允许 用 户 输入 群众 电话 所 反映 事件 的 详细 信息 并 保存 到 数据 库 。 此 
页 面 是 在 12.3.3 节 所 介绍 的 事件 详情 用 户 控件 EventInfo.ascx 基础 上 实现 的 ， 由 于 
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Eventmfo.ascx 控件 实现 了 事件 编辑 和 保存 的 功能 ， 因 此 电话 业务 受理 页 面 代码 较 少 。 创建 
电话 业务 受理 页 面具 体操 作 步 又 如 下 : 

(1) 在 表现 层 项 目 SIL.Web 的 UserControls 文件 夹 中 添加 一 个 事件 编辑 用 户 控 件 
EventEditascx， 此 控件 在 事件 详情 用 户 控 件 EventInfo.ascx 基础 上 添加 了 保存 、 取 消 等 按 
钮 ， 代 码 如 下 : 


<%@ Control Language="C#" AutoEventWireup="true" CodeBehind="EventEdit 
Control.ascx.cs" Inherits="SJL.Web.UserControls.EventEditControl" %> 
<%@ Register src="EventInfoControl.ascx" tagname="EventInfoControl" 
tagprefix="ucl" 各 > 

<asp:Button ID="ok" runat="server" Text=" 保 存 " onclick="ok Click" 
/>gnbsp; gnbsp; 

<input type="button" onclick="window.location=window.location;" value=" 
取消 " />gnbsp; gnbsp; 

<input type="button" onclick="window.location='EventEditPage.aspx?id= 
-1';"” value=" 新 奸 "”/>&nbsp; gnbsp; 

<input type="button" onclick="window.close();"” value=" 关 闭 " /> 

<!-- <input type="button" onclick="alert(' 此 功能 尚未 实现 。' ) ;"” value=" 流 转 " 
/> --> 

<ucl:EventInfoControl ID="eventInfol" runat="server" /> 


(2) 为 EventEdit.ascx 控件 的 几 个 按钮 编写 代码 : 


protected void Page Load (object sender, EventArgs e) 


: 


if (!IsPostBack) 
{ 
//QueryString 中 ID 参数 表示 要 编辑 的 事件 编号 
string s = Request.QueryString["id"]; 
3 (EE 二 // 如 果 为 -1 则 是 添加 新 事件 
eventInfol .newRecord (); 
else 
eventInfol .setByID(s); 
| 
} 
protected void cancel Click(object sender, EventArgs e) 
{ 
eventInfol.theEvent = null; // 单 击 取消 按钮 ， 清 空 控件 内 容 
} 
protected void ok Click(object sender, EventArgs e) 
{ 
// 单 击 确定 按钮 ， 保 存 事件 信息 并 提示 
int n=eventInfol.save(); 
if (n>0) 
Page.ClientScript.RegisterStartupScript (this.GetType(), "succe- 
SS "alert(" 保 存 成 功 1 "7" true)s 
protected void add Click(object sender, EventArgs e) 
{ 
eventInfol .newRecord(); // 单 击 添加 按钮 时 产生 一 条 新 记录 
ok.Enabled = true; 
} 


(3) 在 SJL.Web 项 目 中 添加 一 个 新 页 面 EventEditPage.aspx, 在 页 面 中 放置 一 个 上 一 步 
所 创建 的 EventEdit.ascx 控件 : 


<asp:Content ID="Content2" ContentPlaceHolderID="content" runat="server"> 


和 
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<h3> 电 话 业务 受理 </h3> 
<ucl:EventEditControl ID="EventEditControll" runat="server" /> 
</asp:Content> 


(4) 运行 EventEditPage.aspx， 运 行 界面 如 图 12.8 所 示 。 
BEE oy 


文件 中 


篇 名人) 历史 GE) 书签 虽 工具 四 帮助 0 


WE 


[Ofer /oddost tTr8/PodlicPhone/Defaal Ev rentEaitPsge erp?idr-1 羡 ~ [ea: 


> 区 


|] 电话 业务 受理 


=A 县 书记 县 长 公开 电话 受理 平台 


电话 业务 受理 
保存 取消 新 建 关闭 
事项 编号 受理 日 期 2010-9-4 受理 类 别 [ 害 长 从业 司 受 浊 人 员 管理 员 
反映 人 电话 地 址 请 意 程度 [机 未 证 人 “到 
承办 单位 [一 请 选择 一 到 承办 人 员 承办 单位 2 [二 请 过 反 一 司 条 办 单位 3 [一 请 这 择 一 习 
问题 类 型 [环境 保护 - 司 处 理 状态 [村 办 司 办 结 日 其 签 批 领导 
反映 主题 
反映 内 容 
答复 意见 


当前 日 期 2010 年 9 月 4 日 星期 六 Saturday 当前 用 户 : 管理 员 【退出 登录 】 


-A_ 县 书记 县 长 公开 电话 受理 平台 。 公 开 电 话 : (010) 12345678，87654321。 
主办 单位 ，_B_ 市 -A_ 县 人 民政 府 。 技 术 支持 ，zun. j. 1. studio@enail. com 
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图 12.8 电话 业务 受理 页 面 


电话 业务 综 综合 


在 县 长 公开 电话 受理 系统 中 ， 用 户 可 以 对 所 有 群众 电话 反映 的 事件 进行 各 种 查询 ， 查 
询 条 件 包括 受理 日 期 、 反 映 人 姓名 、 承 办 单位 、 承 办 类 型 等 ， 用 户 可 以 根据 以 上 条 件 中 的 
一 个 或 者 任意 多 个 组 合 进行 查询 ， 并 对 查询 出 的 事件 进行 一 些 操作 ， 如 编辑 、 打 印 等 。 本 
节 将 介绍 电话 业务 综合 查询 功能 的 设计 和 实现 。 


12.4.1 


通用 组 合 条 件 查询 


在 各 种 数据 库 应 用 程序 中 ， 都 会 遇 到 组 合 条 件 查询 ， 即 根据 某 些 条 件 的 任意 组 合 对 某 
一 数据 源 进行 查询 。 在 以 前 的 编程 语言 中 ， 很 难 编写 一 个 强 类 型 的 通用 组 合 查询 方法 。 
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在 NET 4.0 中 ， 利 用 泛 型 、Lambda 表达 式 、LINQ、 实 体 框架 等 技术 ， 就 可 以 设计 出 一 个 


适 上 


于 各 个 数据 实体 的 、 接 受 任意 查询 条 件 且 支持 分 页 功能 的 通用 组 合 条 件 查询 。 下 面 将 


介绍 这 种 查询 的 设计 思路 和 实现 代码 ， 下 面 的 讨论 以 公开 电话 受理 系统 的 事件 查询 为 例 。 


为 了 设计 出 通用 的 组 合 条 件 查询 方法 ， 首 先 要 对 各 种 查询 及 其 实现 进行 分 析 ， 然 后 寻 


找 各 种 查询 的 共同 点 和 不 同 点 , 对 共同 点 进行 抽象 , 不同 点 进行 封装 ， 进 而 实现 通用 设计 。 
下 面 的 讨论 就 是 按照 这 个 思路 进行 的 。 


1. 简单 查询 
最 简单 的 查询 是 查询 所 有 数据 ， 用 代码 描述 如 下 : 


public List<EventInfo> getAll () 
{ 


var context = new UserRightContext (); // 创 建 objectContext 对 象 
context .EventInfo.MergeOption = MergeOption.NoTracking; 
// 不 需要 跟踪 对 象 状态 

var query = from e in context-EventInfo 

select e; // 定 义 查询 
var list = query.ToList() 7 // 执 行 查询 ， 得 到 结果 
context .Dispose(); // 释 放 资源 
return list; // 返 回 结果 


} 


2. 单条 件 查询 
单条 件 查询 可 以 根据 某 一 条 件 进行 查询 ， 如 根据 受理 人 员 、 受 理 日 期 、 事 件 编号 等 进 


行 查询 。 实 现 此 功能 的 代码 如 下 : 


/// <summary> 

/// 根据 受理 人 员 查 询 事 件 详 情 

/// </summary> 

/// <param name="handler"> 受 理 人 员 </param> 

public List<EventInfo> getByHandler (string handler) 
1 


Var context = new UserRightContext (); // 创 建 ObjectContext 对 象 
context .EventInfo.MergeOption = MergeOption.NoTracking; 
// 不 需要 跟踪 对 象 状态 
// 创 建 LINQ 查询 


Var query = from e in context.EventInfo 
where e.HandleUser==handler 


select e; 
var list = query.ToList(); // 执 行 查询 ， 得 到 结果 
context .Dispose(); // 释 放 资 源 
return list; // 返 回 结果 


} 
/// <summary> 
/// 根据 受理 时 间 查 询 事件 详情 
/// </summary> 
///<param name="begin"> 查 询 开始 时 间 </param> 
///<param name="end"> 查 询 结束 时 间 </param> 
public List<EventInfo> getByHandler (DateTime begin, DateTime end) 
| 
var context = new UserRightContext (); // 创 建 ObjectContext 对 象 
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context .EventInfo.MergeOption = MergeOption.NoTracking; 
// 不 需要 跟踪 对 象 状态 
// 定 义 LINQ 查询 
Var query = from e in context.EventInfo 
where e.HandleDate>=begin && e.HandleDate<=end 
select e; 


var list = query.ToList(); // 执 行 查询 ， 得 到 结果 
context.Dispose(); // 释 放 资源 
return list; // 返 回 结果 


上 述 代 码 仅 实现 了 根据 受理 人 和 受理 日 期 进行 查询 的 方法 。 在 实际 项 目 中 ， 要 查询 的 
条 件 远 不 止 这 两 种 ， 用 这 种 思路 实现 查询 ， 每 个 查询 都 需要 写 一 个 方法 ， 则 类 中 会 包含 很 
多 类 似 的 方法 。 

3. 组 合 条 件 查询 


组 合 条 件 查询 是 指 按照 两 个 以 上 的 不 同 条 件 组 合 进行 查询 ， 例 如 查询 受理 人 张 三 在 一 
月 份 受理 的 所 有 事件 ， 或 者 查询 受理 人 张 三 受 理 的 所 有 环境 保护 类 事件 情况 等 ， 示 例 代码 
如 下 : 


public List<EventInfo> getByHandlerAndDate (string handler, DateTime begin, 
DateTime end) 


{ 


var context=new UserRightContext (); 
context.EventInfo.MergeOption = MergeOption.NoTracking; 
// 定 义 LINQ 查询 
Var query=from e in context.EventInfo 

where e.HandleDate>=begin 

&& e.HandleDate<=end 

&& e.HandleUser==handler 


select e; 
var list = query.ToList(); / /执行 查询 ， 得 到 结果 
context .Dispose(); // 释 放 资 源 


return list; 


; 

如 果 查 询 条 件 共 有 NN 种 ， 那 么 各 种 条 件 组 合 可 达到 N 的 阶乘 ， 按 照 这 种 思路 写 代码 ， 
要 想 实 现 所 有 条 件 组 合 几乎 是 不 可 能 的 。 

4. 分 页 查询 

在 真实 项 目 中 ， 由 于 数据 量 巨大 ， 绝 大 多 数 查 询 都 是 分 页 查询 ， 每 次 仅 需 返 回 指定 页 
码 的 数据 ， 示 例 代 码 如 下 : 


public List<EventInfo> pageQuery() 
人 


int pageIndex = 3; // 页 码 

int pageSize = 10; // 页 面 大 小 

Var context = new UserRightContext (); 

context .EventInfo.MergeOption = MergeOption.NoTracking; 

Var query = from e in context.EventInfo 
where e.HandleDate >= DateTime.Parse ("2010-1-1") 
&& e.HandleDate <= DateTime.Parse("2010-1-31 23:59:59") 
&& e-HandleUser == "admin" 
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orderby e-EventID // 排 序 
select e; 
var list = query.Skip( (pageIndex-1)*pageSize) -Take (pageSize). 
ToList () ;// 执 行 分 页 查询 
context .Dispose(); // 释 放 资 源 
return list; 


} 


综合 分 析 上 述 各 种 查询 ， 可 以 得 出 以 下 结论 : 

(1) 各 种 查询 的 共同 点 是 都 需要 创建 ObjectContext， 都 不 用 跟踪 对 象 状 态 ， 都 返回 一 
个 实体 列表 ， 都 需要 分 页 〈 不 分 页 可 视 为 页 面 大 小 极 大 的 特殊 情况 )。 

(2) 各 种 查询 的 不 同 点 是 返回 的 实体 类 型 不 同 ， 用 到 的 ObjectQuery 查询 对 象 不 同 ， 
查询 条件 不 固定 ， 排 序 方法 不 一 样 。 

如 果 可 以 编写 一 个 查询 方法 ， 此 方法 可 以 指定 查询 所 使 用 的 ObjectQuery 类 型 、 返 回 
的 实体 类 型 、 任 意 数量 的 查询 条 件 、 排 序 关 键 字 和 升降 序 ， 则 可 以 实现 通用 组 合 条 件 查询 。 
其 中 类 型 参数 可 用 泛 型 实现 ,查询 条 件 可 用 Lambda 表达 式 实现 ， 排 序 表达 式 也 用 Lambda 
表达 式 实 现 。 按 照 这 个 思路 ， 编 写 通用 组 合 条 件 查询 方法 如 下 : 

/// <summary> 

/// 通用 组 合 条 件 查 询 

/// </summary> 

/// <typeparam name="T"> 实 体 对 象 类 型 </typeparam> 

/// <typeparam name="TSortKey"> 排 序 关键 字 类 型 </typeparam> 

/// <param name="query">ObjectQuery 对 象 </param> 

/// <param name="sort"> 排 序 表达 式 </param> 

/// <param name="sortDesc"> 是 否 降序 排序 </param> 

/// <param name="page"> 分 页 参数 </param> 

/// <param name="dispose"> 查 询 结束 是 否 释 放 ObjectContext 资源 </param> 

/// <param name="predicate"> 查 询 条 件 </param> 

/// <returns> 查 询 结果 列表 </returns> 

internal static List<T> selectMany<T,TSortKey> (ObjectQuery<T> query, 

Expression<Func<T, TSortKey>> sort, bool sortDesc, 

PageDataArgument page, bool dispose, 


params Expression<Func<T,bool>>[] predicate) /1/0 到 多 个 查询 条 件 
where T:EntityObject 


query.MergeOption = MergeOption.NoTracking; // 不 需要 跟踪 对 象 状态 
IQueryable<T> q = query; / /查询 对 象 
for (int i = 0; i < predicate.Length; i++) / /逐个 添加 查询 条 件 
{ 

q = q.Where (predicate[i]); 
} 


if (sortDesc) // 添 加 排序 条 件 
q = q.OrderByDescending (sort); 

else 
q = q.OrderBy (sort); 

if (page.refreshCount) // 刷 新 记录 总 数 


page -count = q.Count (); 
Var list = q-.Skip(page.pageIndex * page.pageSize) .Take (page.pageSize). 
ToList(); 
detachEntity (query.Context, list); // 分 离 实体 对 象 
if (dispose) 
query.Context .Dispose(); 
return list; 
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} 
/// <summary> 
/// 将 列表 中 的 实体 对 象 从 ObjectContext 中 分 离 
/// </summary> 
/// <typeparam name="T"> 实 体 对 象 类 型 </typeparam> 
/// <param name="context">ObjectContext</param> 
/// <param name="]ist"> 对 象 列表 </param> 
internal static void detachEntity<T> 

(ObjectContext context, List<T> list) 

where T : EntityObject 
{ 

list.ForEach (item =>detachAndRemoveEntityKey (context, item)); 
} 
// 从 对 象 上 下 文 分 享 对 象 并 删除 实体 键 
private static void detachAndRemoveEntityKey (ObjectContext context, 
EntityObject entity) 
{ 

if(entity.EntityState!=EntityState.Detached) 

context .Detach (entity); 
entity.EntityKey = null; 


全 提示 : selectMany0 方 法 的 最 后 一 个 参数 前 有 一 个 params 关键 字 ， 表 示 这 是 数目 可 变 的 
参数 ， 即 可 以 接受 0 到 多 个 查询 条 件 。 


上 述 代 码 中 selectMany0 方 法 参数 太 多 ， 调 用 不 方便 且 不 易 阅 读 。 可 以 把 这 些 参数 封 
装 到 一 个 类 ObjectQueryArgument 中 ， 为 各 个 参数 设置 默认 值 ， 这 样 就 能 够 使 selectMany() 
方法 的 使 用 更 加 简洁 。ObjectQueryArgument 类 代码 如 下 : 


/// <summary> 
/// 通用 组 合 查询 参数 ， 在 EntityUtility.selectMany () 方 法 中 使 用 
/// </summary> 
/// <typeparam name="T"> 实 体 对 象 类 型 </typeparam> 
/// <typeparam name="TSortKey"> 排 序 关键 字 类 型 </typeparam> 
internal class ObjectQueryArgument<T,TSortKey> 
where T:IEntityWithKey 
{ 
#region 构造 函数 
internal ObjectQueryArgument () 
disposeAfterQuery = true; 
descending = false; 
} 
internal ObjectQueryArgument (ObjectQuery<T> theQuery, Expression 
<Func<T, TSortKey>> sortExp,PageDataArgument pageArg) 
{ 
descending=false; 
query=theQuery; 
sort=sortExp; 
page=pageArg; 
} 
#endregion 
#region 属性 
internal ObjectQuery<T> query { get; set; } 
internal Expression<Func<T, TSortKey>> sort{get;set;} 
internal bool descending { get; set; } 
internal PageDataArgument page { get; set; } 
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// 执 行 完 查询 后 是 否 释 放 ObjectContext 
internal bool disposeAfterQuery { get; set; } 
#endregion 


} 
使 用 ObjectQueryArgument 类 作为 参数 的 selectMany0 方 法 如 下 : 


internal static List<T> selectMany<T, TSortKey> 


(ObjectQueryArgument<T, TSortKey> arg, // 查 询 参数 
params Expression<Func<T, bool>>[] predicate) // 查 询 条 件 
where T:EntityObject // 泛 型 约束 


return selectMany<T, TSortKey> 
(arg.query, arg.sort, arg.descending, arg.page, arg.disposeAfter-— 
Query, predicate); 


12.4.2 ”电话 业务 综合 查询 数据 层 和 业务 层 


通过 与 用 户 交流 ， 电 话 业务 查询 主要 根据 以 下 几 个 条 件 : 受理 日 期 、 事 件 编号 、 反 映 


人 姓名 、 责 任 单位 、 事 件 类 型 、 事 件 来 源 。 基 于 12.4.1 节 所 设计 的 通用 组 合 条件 查 询 方法 ， 
能 够 实现 电话 业务 综合 查询 功能 ， 有 具体 实现 步骤 如 下 : 


(1) 在 数据 访问 层 项 目 SIL.Dal 的 EventInfoDal 类 中 ， 添 加 一 个 search0 方 法 ， 实 现 综 


合 查询 功能 ， 代 码 如 下 ; 


/// <summary> 

/// 综合 查询 

/// </summary> 

/// <param name="page"> 分 页 参数 </param> 

/// <param name="predicates"> 查 询 条件 </param> 

/// <returns> 查 询 结 果 </returns> 

public static List<EventInfo> search (PageDataArgument page, 
params Expression<Func<EventInfo,bool>>[] predicates) 


. 
Var arg = new ObjectQueryArgument<EventInfo, DateTime?>(); 
arg.query = getQuery(); 


arg.sort = e => e.HandleDate; // 按 照 受理 日 期 排序 
arg.descending = true; /7 降序 排序 
arg.disposeAfterQuery = false; // 不 立即 释放 资源 
var list= EntityUtility.selectMany (arg,predicates);// 执 行 查询 
list.ForEach (item => loadDetail (item)); // 加 载 外 键 数据 
arg.query.Context .Dispose(); // 释 放 资源 


return list; 


(2) 在 EventInfoDal 类 中 添加 一 个 loadReference() 方 法 ， 用 于 加 载 事件 信息 的 外 键 数 
如 事件 来 源 、 责 任 部 门 等 

/// <summary> 

/// 加 载 事件 外 键 数据 

/// </summary> 

/// <param name="info"> 要 加 载 的 事件 </param> 

private static void loadReference (EventInfo info) 


有 
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if (info.EventSource.HasValue) 
info .eventSourceName = info.EventSourcel .Name; 

if (info.TypeID.HasValue) 
info.typeName=info.EventType.Name; 

if (info.StatusID.HasValue) 
info.statusName = info.EventStatus.Name; 

if (!string.IsNullOrEmpty (info.ResponsibleDepartment)) 


info.responsibleDepartmentName = info.Department .DepartmentName; 


info.DepartmentlReference.Load(); 
info.Department2Reference.Load(); 


(3) 目前 为 止 ， 实现 的 综合 查询 search() 方 法 需要 传递 多 个 Lamba 表达 式 ， 方 法 调用 


不 够 简洁 ， 可 以 添加 一 个 重 载 的 search0 方 法 用 更 清晰 、 直 观 的 方式 实现 同样 的 查询 功能 。 
为 此 ， 需 要 添加 一 个 类 EventSearchCondition 封装 事件 综合 查询 条 件 ， 然 后 将 此 类 作为 参 


数 传 递 给 search0 方 法 。 代 码 如 下 : 


/// <summary> 
/// 事件 查询 条 件 
/// </summary> 
public class EventSearchCondition 
{ 
public string id { get; set; } 
public int source { get; set; } 
public int type { get; set; } 
public string department { get; set; } 
public string person { get; set; } 
public DateRange date { get; set; } 
} 
/// <summary> 
/// 综合 查询 
/// </summary> 
/// <param name="condition"> 查 询 条 件 </param> 
/// <param name="page"> 分 页 参数 </param> 
/// <returns> 查 询 结 果 </returns> 
public static List<EventInfo> search (EventSearchCondition 
PageDataArgument page) 
{ 


List<Expression<Func<EventInfo, bool>>> predicates 
= new List<Expression<Func<EventInfo, bool>>>(); 
// 如 果 时 间 范 围 不 为 空 ， 则 添加 这 个 查询 条 件 
if (condition.date != null) 
predicates.Add( 
e => e.HandleDate >= condition.date.from 
&& e.HandleDate < condition.date.to); 
// 如 果 责 任 部 门 不 为 空 ， 则 添加 此 条 件 
if (!string.IsNullOrEmpty (condition.department)) 


// 事 件 编号 
// 事 件 来 源 
// 事 件 类 型 
// 责 任 部 门 
/1 反映 人 姓名 
// 日 期 范围 


condition, 


// 查 询 条 件 列表 


predicates.Add (e => e.ResponsibleDepartment == condition. 


department); 


// 如 果 反 映 人 不 为 空 ， 则 添加 此 条 件 
if (!string.IsNullOrEmpty (condition.person)) 


predicates.Add(e => e.PersonName == condition.person); 


// 如 果 事 件 来 源 不 为 空 ， 则 添加 此 条 件 


if(condition.source>0) 


predicates.Add(e => e.EventSource == condition.source); 


// 如 果 事 件 类 型 不 为 空 ， 则 添加 此 条 件 
if(condition.type>0) 
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} 


predicates.Add(e => e.TypelID == condition.type); 
if (predicates.Count == 0) 

return search(page); 
return search(page, predicates.ToArray()); 


(4) 在 业务 逻辑 层 项 目 SJL.Bll 的 EventmfoBll 类 中 ， 添 加 一 个 search0 方 法 ， 代 码 


如 下 : 


/// <summary> 

/// 综合 查询 

/// </summary> 

/// <param name="condition"> 查 询 条 件 </param> 

/// <param name="page"> 分 页 参数 </param> 

/// <returns> 查 询 结 果 </returns> 

public static List<EventInfo> search (EventSearchCondition condition, 
PageDataArgument page) 


{ 


/ /如果 事件 编号 不 为 空 ， 则 只 根据 事件 编号 查询 (忽略 其 他 条 件 ) 
if (!string.IsNullOrEmpty (condition.id)) 


{ 
List<EventInfo> list=new List<EventInfo>(); 
Var item=getByID(condition.id); 
if (item!=null) 
list.Add (item); 
return list; 
} 
return DB.search (condition, page); // 执 行 综合 查询 


12.4.3 事件 列表 控件 


在 公开 电话 受理 系统 中 有 多 个 需要 显示 事件 信息 的 列表 ， 为 了 实现 界面 和 代码 复 用 ， 


可 以 月 


日 一 个 用 户 控 件 EventListControl.ascx 封装 事件 列表 功能 ,然后 在 需要 显示 事件 列表 的 


页 面 上 直接 使 用 此 用 户 控 件 。 具 体 实现 步 又 如 下 : 
(1) 在 表现 层 项 目 SJL.Web 的 UserControls 文件 夹 中 添加 一 个 EventListControl.ascx 


用 户 控 件 。 


(2) 在 用 户 控件 EventListControl.ascx 中 放置 一 个 GridView， 以 列表 形式 显示 事件 主 
要 信息 ， 并 在 GridView 最 后 几 列 添加 编辑 和 打印 按钮 ， 如 图 12.9 所 示 。 


受理 编号 受理 日 其 反映 人 受理 未 源 受理 关于 承办 单位 
Datahoumd 。 Tatabound Databourd Databourd Databomd Databound Databound 办 理 承办 反馈 
Databoud 。 Tatabound Databourd Databourd Databomd Databound Databhound 办 理 承办 反馈 
Databound 。 Databound Databourd Databourd Databound Databound Databound 办 理 承 失 反馈 


Databound 。 Tatabound Databourd Databourd Datsbomd staba nc 办 理 承办 反馈 


Databound 。 Tatabound Databourd Databourd Databauma at abourd Databen 夫 理 承办 反馈 


OSplit | 加 Soaree 


图 12.9 事件 列表 用 户 控件 


户 控 件 EventListControl.ascx 代码 如 下 : 


<%@ Control Language="C#" AutoEventWireup="true" CodeBehind="EventList-— 


。405 。 


第 3 篇 _ 项 目 实战 


Control.ascx.cs" Inherits="SJL.Web.UserControls.EventListControl" $%> 
<script type="text/javascript"> 
// 打 开 编 辑 对 话 框 
function openEditDialog(id) { 
openMyDialog("../Default/EventEditSmallPage.aspx", "?id=" + id); 
} 
// 打 开打 印 对 话 框 
function openPrintDialog(id) { 
window.showModalDialog("../Default/EventPrintPage.aspx"+"?id=" + 
dj 
} 
// 打 开 一 个 浏览 器 模式 对 话 框 
function openMyDialog (url, querystring) { 
window.showModalDialog (url+querystring, null, 
"dialogWidth:960; dialogHeight:500; center:yes; menubar:yes;"); 
} 
</script> 


<asp:GridView ID="gridl" CssClass="gridview" runat="server" 
DataKeyNames="EventID" AutoGenerateColumns="false" > 

<Columns> 

<asp:BoundField DataField="EventID" HeaderText=" 受 理 编号 " ItemStyle-Width= 

BO /> 

<asp:BoundField DataField="HandleDate" DataFormatSstring="{0:d}" Header- 

Text=" 受 理 日 期 " ItemStyle-Width="100" /> 

<asp:BoundField DataField="PersonName" HeaderText=" 反 映 人 " ItemStyle- 

Width="70"/> 

<asp:BoundField DataField="EventSourceName" HeaderText=" 受 理 来 源 " ItemSty- 

le-Width="100" /> 

<asp:BoundField DataField="TypeName" HeaderText=" 受 理 类 型 " ItemStyle- 


<asp:BoundField DataField="ResponsibleDepartmentName" HeaderText=" 承 办 单位 
" ItemStyle-Width="120" /> 
<asp:BoundField DataField="StatusName" HeaderText=" 处 理 情况 " ItemStyle- 
Width="60" /> 
<asp:BoundField DataField="EvaluationName" HeaderText=" 满 意 度 " ItemStyle- 
Width="80" /> 
<asp:TemplateField HeaderText=" 详 情 "> 
<ItemTemplate> 
<a href="#" onclick="openEditDialog('<%#Eval ("EventID")$>');" > <%-- 编 辑 
链接 --$> 
<img src="../images/edit.png" alt="" class="noborder"/> </a> 
</ItemTemplate> 
</asp:TemplateField> 
<asp:TemplateField HeaderText=" 打 印 单据 "> <% 一 -打印 链接 --%> 
<ItemTemplate> 
<a href="../Default/BanLiPrintPage.aspx?id=<%#Eval ("EventID")®%>" target= 
"eventPrintWindow"” > 办 理 </a> 
<a href="../Default/CBPrintPage.aspx?id=<%#Eval ("EventID")%®>" target= 
"eventPrintWindow" > 承办 </a> 
<a href="../Default/FKPrintPage.aspx?id=<%#Eval ("EventID")®$>" target= 
"eventPrintWindow" > 反馈 </a> 
</ItemTemplate> 
</asp:TemplateField> 
</Columns> 

</asp:GridView> 


(3) 在 用 户 控件 EventListControl ascx.cs 中 编写 一 个 方法 ， 将 数据 绑 定 到 GridView: 
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public void bindList (List<EventInfo> 1ist) 
{ 


gridl.DataSource = list; 
gridl.DataBind(); 


12.4.4 综合 查询 页 面 


在 综合 查询 页 面 中 ， 用 户 可 以 输入 各 种 查询 条 件 并 执行 查询 ， 还 可 以 对 查询 结果 进行 


编辑 、 打 印 等 操作 。 实 现 综合 查询 页 面 的 具体 操作 步骤 如 下 : 


(1) 在 表现 层 项 目 SIL.Web 中 ， 从 母 版 页 SjlMastermaster 创建 一 个 新 的 页 面 


EventListPage.aspx， 在 页 面 上 放置 多 个 文本 框 控件 以 允许 用 户 输入 查询 条 件 ， 使 
UI 中 的 日 历 控件 输入 日 期 ， 放 置 一 个 EventListControl 用 户 控件 以 显示 查询 结果 ， 
码 所 示 ; 


jQuery 
如 下 代 


<asp:Content ID="Contentl" ContentPlaceHolderID="head" runat="server"> 


<script type="text/javascript"> 
$(function() { 
initControls(); 
Ws //$ (function) 
function initControls() { 
// 为 GridView 添加 光 棒 效果 
$('table.gridview') .find("tr") .hover( 
function() { $(this).addClass('hoverRow'); }, 
function() { $(this) .removeClass('hoverRow'); } 
); //$('table') .tr.hover 
$SjlUtility.addButtonClass (); // 为 按钮 添加 样式 
$SjlUtility.jQueryDatePickerChinese(); 
// 初 始 化 jQuery UI 日历 控 件 
$('#<%=datel .ClientID%>') .datepicker (); 


// 为 datel 控件 启用 jQuery UI 
$('#<%=date2.ClientID%>') .datepicker (); 

// 为 date2 控件 启用 jQuery UI 
$('#printbutton') .click( printGrid ); 

// 为 打印 按钮 添加 事件 处 理 程序 


} 
function printGrid() { 
window.open ("EventListPrint.aspx"); 
} 
</script> 
</asp:Content> 
<asp:Content ID="Content2" ContentPlaceHolderID="content" runat="se. 


<h3> 受 理 业 务 列表 </h3> 


rver"> 


事件 编号 : <asp:TextBox ID="eventID" runat="server"></asp:TextBox> 


受理 日 期 : <asp:TextBox ID="datel" runat="server" ></asp:TextBox> 
至 <asp:TextBox ID="date2" runat="server"></asp:TextBox> 

事件 来 源 : <asp:DropDownList ID="sourceList" runat="server"></asp 
DropDownList><br /> 


承办 类 型 : <asp:DropDownList ID="typeList" runat="server"></asp: Drop- 


DownList> 
承办 单位 : <asp:DropDownList ID="departmentList" runat="server" 


DataValueField="departmentID" DataTextField="DepartmentName"></asp: 


DropDownList> 
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反映 人 : <asp:TextBox ID="person" runat="server"></asp:TextBox> &nbsp; 
<asp:Button ID="clear" runat="server" Text=" 清 空 条 件 " onc1lick="clear 
ER 
gnbsp; <asp:Button ID="Buttonl" runat="server"” Text=" 执 行 查询 " onclick= 
"Buttonl Click" /> 
&nbsp; <input type="button" value=" 打 印 表格 " id="printbutton" /> 
<ucl:EventListControl ID="listl" runat="server" /> 
<webdiyer:AspNetPager ID="pagerl" runat="server" 
onpagechanged="pagerl PageChanged" PageSize="20"> 

</webdiyer:AspNetPager> 

</asp:Content> 


(2) 在 EventListPage.aspx 页 面 的 Page_Load 事件 中 将 数据 绑 定 到 查询 条 件 中 的 下 拉 列 
表 中 ， 包 括 责任 单位 列表 、 事 件 来 源 列表 和 事件 类 型 列表 : 


protected void Page Load (object sender, EventArgs e) 


{ 
if (!IsPostBack) 


{ 
» 


initList(); 


// 绑 定 过 滤 条 件 中 的 下 拉 列 表 
private void initList() 
{ 
string all="-- 全 部 --"; 
// 绑 定 事件 来 源 列表 
WebUtility.bindListWithDictionary<EventSourceB11， 
EventSource> (sourceList); 
typeList.Items.Insert (0,new ListItem(all,"-1")); 
// 绑 定 事件 类 型 列表 
WebUtility.bindListwithDictionary<EventTYPeB11，EventType> (typeList) 
sourceList.Items.Insert (0,new ListItem(all,"-1")); 
// 绑 定 责任 单位 列表 
var list = DepartmentBLL .getRll (PageDataRArgument .allData); 
departmentList.DataSource = list; 
departmentList.DataBind(); 
departmentList.Items.Insert (0,new ListItem(all, "")); 


(3) 在 “执行 查询 ”按钮 的 Click 事件 中 ， 根 据 查 询 条 件 执行 查询 并 显示 结果 : 


protected void Buttonl Click(object sender, EventArgs e) 
{ 

pagerl .CurrentPagelIndex = 1; 

bindData (); 
} 


private void bindData() 
{ 
Var list = executeQuery() 
list1.bindList (list); 
} 
/// <summary> 
/// 根据 用 户 输入 的 查询 条 件 执行 查询 
/// </summary> 
/// <returns> 查 询 结 果 </returns> 
private List<SJL.Entity.EventInfo> executeQuery() 


{ 
PageDataArgument page= new PageDataArgument (pagerl .CurrentPage 


。408 。 


第 12 章 县 长 公开 电话 受理 系统 


Index = 1, 

pagerl .PageSize, true); 
// 以 下 语句 设置 查询 条 件 
EventSearchCondition condition = new EventSearchCondition(); 
condition.id = eventID.Text; 
condition.date = DateRange.between2Date (datel .Text, date2.Text); 
condition.department = departmentList.SelectedValue; 
condition.person = person.Text; 
condition.source = int.Parse(sourceList.SelectedValue); 
condition.type = int.Parse (typeList.SelectedValue); 
var list= EventInfoBLL.search (condition, page); // 执 行 查询 
Session["EventListPrintData"] = list; 
pagerl]l .RecordCount = page.count; 
return list; 


3 
(4) 在 分 页 控件 的 PageChanged 事件 中 重新 绑 定 新 页 面 数据 : 


protected void pagerl PageChanged (object sender, EventArgs e) 


{ 
bindData (); 
; 
(5) 在 浏览 器 中 查看 EventListPage.aspx 页 面 ， 运 行 界面 如 图 12.10 所 示 。 
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图 12.10 电话 业务 综合 查询 页 面 


12.5 报表 打印 


在 县 长 公开 电话 受理 系统 中 ， 需 要 产生 各 种 单据 和 表格 ， 这 些 单据 和 表格 需要 打印 出 
来 ， 让 相关 人 员 浏 览 、 签 字 并 存档 。 本 节 将 介绍 如 何 生成 和 打印 这 些 表格 。 


12.5.1 报表 母 版 页 


通过 对 县 长 公开 电话 受理 系统 中 所 有 表格 进行 分 析 ， 发 现 各 种 单据 和 表格 拥有 一 些 共 
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同 点 : 都 需要 打印 到 A4 纸张 上 ， 都 有 一 个 表 头 ， 表 格 字体 大 小 要 求 可 以 调整 。 为 了 实现 


复 上 
页 上 


， 可 以 设计 一 个 专用 于 打印 报表 的 母 版 页 ， 把 这 些 通 用 功能 放 在 母 版 页 中 。 报 表 母 版 
的 具体 实现 步骤 如 下 : 
(1) 在 公开 电话 受理 系统 的 表现 层 项 目 SIL.Web 中 添加 一 个 新 的 母 版 页 ， 命 名 为 


ReportMastermaster。 


(2) 在 母 版 页 上 放置 一 个 div， 用 CSS 将 其 大 小 、 边 距 等 设置 为 与 A4 纸张 相 适 应 : 


<style type="text/css"> 

.a4{ margin-left:1l0px; margin-top:10px; width:640px; font-size:16px; } 
</style> 

<div class="a4"> 

</div> 


(3) 在 页 面 上 放置 一 个 Label 以 显示 报表 标题 , 放置 一 个 ContentPlaceHolder 以 在 内 容 


页 中 显示 具体 报表 内 容 。 


“4 


<div class="a4"> 
<div id="thetitle" class="reporttitle"> 
<asp:Label ID="reportTitle" runat="server"” Text=" 书 记 、 县 长 公开 电话 受理 事 
项 承办 单 "> </asp:Label> 
</div> 
<asp:ContentPlaceHolder ID="content" runat="server"> 
</asp:ContentPlaceHolder> 
</div> 
</div> 


(4) 在 母 版 页 顶部 放置 用 于 打印 、 设 置 表 头 和 调整 字体 大 小 的 控件 : 


<div class="noprint"> 

<input type="button" value=" 打 印 " onclick="window.print();" /> 

<input type="button" value=" 编 辑 " id="editButton" /> 

表 头 : <input type="text" id="titleText" style="width:300px;"” value=" 书 记 、 
县 长 公开 电话 受理 事项 XX 单 " /> 

<input type="button" value=" 设 置 表 头 " id="titleButton" /> 

&nbsp; 字 体 大 小 : <select id="fontSizeOption" style="width:auto;"> 

<option value="12">1</option> 

<option value="14">2</option> 

<option value="16" selected>3</option> 

<option value="18">4</option> 

<option value="20">5</option> 

</select> 

<br /> 

<span class="obvious"> 提 示 : 如 需 设置 打印 比例 、 页 边 距 等 ， 请 从 浏览 器 相应 菜单 中 进行 操 
作 。</span> 

</div> 


报表 母 版 页 设计 界面 如 图 12.11 所 示 。 
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书记 、 县 长 公开 电话 受理 事项 承办 单 


图 12.11 报表 母 版 页 设计 界面 
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(5) 母 版 页 顶部 的 按钮 、 文 本 框 等 控件 仅 在 浏览 页 面 时 可 见 ， 在 打印 时 不 可 见 ， 可 以 
通过 CSS 实现 这 个 效果 ，CSS 代码 如 下 : 


<style type="text/css" media="print"> // 此 样式 仅 对 打印 机 有 效 
-noprint{display:none;} // 打 印 时 不 显示 
</style> 


(6) 报表 母 版 页 中 的 打印 、 设 置 表 头 、 修 改 字体 等 功能 都 可 以 在 浏览 器 端 实现 而 无 须 
发 回 服务 器 进行 处 理 ， 所 以 为 相应 控件 添加 JavaScript 脚本 以 实现 这 些 功能 : 


<script type="text/javascript"> 
$ (function() { 


$('#editButton') .click (openEditDialog); // 打 开 编 辑 对 话 框 
$('#titleButton') .click (setTitle); // 设 置 报表 标题 
$('"#fontSizeOption') .change (changeFont); // 改 变 字体 大 小 
]) 7 
// 设 置 报表 标题 


function setTitle() { 
$('div#thetitle > span') .html ($('#titleText') .val ()); 


}; 

// 打 开 编 辑 对 话 框 

function openEditDialog() { 
var id = $('#hiddenid') .val (); 
window.showModalDialog("../Default/EventEditSsmallPage.aspx?id=" + 
dy 
null, "dialogWidth:900;dialogHeight:500;center:yes;menubar: 
yes;"); 
window.location = window.location; 


}; 
/ /改变 字体 大 小 
function changeFont() { 
var size = $(this) .val(); 
$('div.a4') .css("font-size", sizet'px"'); 
}; 
</script> 


12.5.2 ”打印 承办 单 


县 长 公开 电话 受理 系统 中 的 承办 单反 映 了 群众 所 反映 事件 的 承办 情况 ， 此 单据 需 将 事 
件 的 基本 信息 打印 出 来 ， 交 由 相关 领导 签字 。 生 成 和 打印 承办 单 的 具体 步骤 如 下 : 
(1) 在 表现 层 项 目 SIL.Web 中 ， 基 于 报表 母 版 页 ReportMastermaster 添加 一 个 新 页 面 
CBPrntPage.aspx。 
(2) 在 页 面 上 放置 一 个 table， 并 按照 用 户 提供 的 样 表 设 计 表 格 的 内 容 和 格式 。 页 面 代 
人 码 如 下 : 
<asp:Content ID="content2" ContentPlaceHolderID="content" runat="server"> 
<input type="hidden" id="hiddenid" value="<%=eventId.Text%>" /> <%-- 事 件 
= 
<table style="border:none;" width="100%" rules="none" > 
<tr> 
<td> 事 项 编号 : 


<asp:Label runat="server" ID="eventId" Text="" ReadOnly="true"></asp: 
Label></td> 


:411: 


第 3 篇 _ 项 目 实战 


<td> 办 理 时 限 : 10 个 工作 日 </td> 
<td> 受理 日 期 : 
<asp:Label ID="theDate" runat="server" Text="" ReadOnly="true"> 
</asp: Label></td> 
Er 
</table> 
<br /> 
<!-- 表 格 总 高 度 : 750px, 不 含 表 头 、 表 尾 和 第 1 行 --> 
<table rules="all" border="1l" width="100%" cellpadding="4" style= 
"text-align:center;"> 
<tr><td class="colheader" style="height:60px;"> 来 电 <br /> 
人 员 </td><td> 
<asp:Label ID="personName" runat="server"></asp:Label> 
</td> 
<td class="colheader"> 联 系 <br /> 
电话 </td><td> 
<asp:Label ID="personPhone" runat="server"></asp:Label> 
</td><td class="colheader"> 
联系 <br /> 
地 址 </td><td> 
<asp:Label ID="personAddress" runat="server"></asp:Label> 
</td> </tr> 


<tr> <td> 内 <br /> 容 <br /> 提 <br /> 要 <br /></td> 
<td colspan="5" style="height:200px; vertical-align:top; text-align: 
Toft;"> <br /> 


<asp:Label ID="eventContent" runat="server" /> 
</td> </tr> 
<tr> <td> 
<!-- 使 用 pre 标记 实现 预 排 版 功能 ， 为 了 对 齐 格式 ， 下 面 的 8 个 字 需 要 显示 在 4 行 --> 
<pre style="font-size:16px;"> 
县 室 
政 领 
府 导 
办 意 
公 见 
</pre> 
</td><td colspan="5" style="height:180px; vertical-align:top; text-— 
align: left;"> </td> 
/ES 
<tr> <td > 领 <br /> 导 <br /> 批 <cbr /> 示 </td> 
<td colspan="5" style="height:250px;"> 
</td> </Er> 
<tr><td> 办 <br /> 理 <br /> 结 <br /> 果 </td><td colspan="5" style="height: 
120pxa > 
</td> </tr> 


</table> 

<div style="margin-right:20px; text-align:right;"> 

受理 人 : <asp:Label runat="server" ID="personList2" Width="120" /> 
初审 人 : <asp:Label runat="server" ID="label01" Width="120" /> 
审核 人 : <asp:Label runat="server" ID="label102" Width="120" /> 
</div> 

</asp:Content> 


(3) 在 页 面 的 Load 事件 中 ， 设 置 报表 标题 和 报表 内 容 。 由 于 报表 标题 控件 在 母 版 页 
中 ， 所 以 要 使 用 MasterFindControl 找到 标题 所 对 应 的 Label 控件 : 


protected void Page Load(object sender, EventArgs e) 
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if (!IsPostBack) 
Label 1=this.Master.FindControl ("reportTitle") as Label; 

// 找 到 母 版 页 中 的 标题 控件 
1.Text = "书记 、 县 长 公开 电话 受理 事项 承办 单 "; // 设 置 标题 文字 
string s = Request.QueryString["id"]; // 得 到 要 显示 的 事件 ID 
setByID(s); 

} 
} 
/// <summary> 
/// 根据 事件 ID 设置 报表 内 容 
/// </summary> 
/// <param name="id"> 事 件 ID</param> 
private void setByID(string id) 
if (string.IsNullOrEmpty (id)) 
return; 
EventInfo info = EventInfoBLL.getByID (id); // 根 据 ID 得 到 事件 详情 
eventId.Text = info.EventID; 
if (info.HandleDate.HasValue) 
theDate.Text = info.HandleDate.Value.ToShortDateString () 7 
// 根 据 事件 详情 设置 各 个 控件 的 值 
PersonName .Text = info.PersonName; 
eventContent .Text = info.EventContent.Replace("\n", "<br/>"); 
personList2.Text = info.HandleUser; 
personAddress.Text = info.PersonAddress; 
personPhone.Text = info.PersonPhone; 


} 
(4) 运行 承办 单 页 面 CBPrintPage.aspx， 运 行 页 面 如 图 12.12 所 示 。 
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图 12.12 承办 单 页 面 
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在 图 12.12 所 示 的 承办 单 页 面 中 单 击 “ 打 印 ”按钮 ， 弹 出 如 图 12.13 所 示 的 “打印 ” 
对 话 框 。 


12.13 浏览 器 “打印 ”对 话 杠 


12.6 小 结 


本 章 介绍 了 A 县 县 长 公开 电话 受理 系统 的 设计 与 实现 , 首先 介绍 了 用 户 需 求 和 系统 整 
体 设计 思路 ， 然 后 重点 讲解 了 业务 受理 、 综 合 查询 和 报表 打印 等 功能 的 设计 和 实现 。 本 系 
统 规模 较 小 ， 是 学 习 ASP.NET 的 一 个 好 案例 。 本 系统 已 经 在 滨州 市 A 县 得 到 成 功 应 用 ， 
对 于 类 似 系统 的 开发 具有 一 定 借鉴 意义 。 
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本 章 所 介绍 的 案例 是 作者 为 某 市 人 力 资源 和 社会 保障 局 开发 的 一 个 软件 。 出 于 安全 和 
隐私 方面 的 原因 ， 书 中 不 便 给 出 该 市 的 真实 名 称 ， 下 文 均 以 A 市 代替 。 本 系统 以 社保 卡 开 
户 银行 的 数据 为 基础 ， 结 合 A 市 人 力 资源 和 社会 保障 局 社保 管理 系统 的 数据 ， 对 社保 卡 消 
费 数据 进行 各 种 处 理 ， 可 以 实现 A 市 社保 卡 的 资金 结算 和 数据 统计 功能 。 


13.1 整体 设计 思 


本 节 将 介绍 A 市 社保 卡 结算 系统 的 开发 背景 和 业务 流程 ， 分 析 用 户 需求 ， 提 出 此 系统 
整体 设计 思路 ， 实 现 数据 库 结构 设计 。 


13.1.1 项 目 简介 


A 市 在 全 市 范围 内 推行 了 社会 保障 卡 发 行 工 作 ， 参 保 人 员 持 社会 保障 卡 即 可 在 全 市 统 
一 定点 的 医疗 机 构 网 络 内 无 障碍 就 医 、 就 诊 和 结算 ,实现 全 市 医疗 服务 “一 卡通 ”。 社 会 保 
障 卡 是 按照 劳动 和 社会 保障 统一 规划 ， 由 各 地 劳动 社会 保障 部 门面 向 城镇 从 业 人 员 和 离 退 
休 和 员 发 行 的 用 于 劳动 和 社会 保障 业务 的 集成 电路 卡 。 社 保 卡 的 发 放 范围 为 全 市 城镇 职工 
医疗 、 养 老 、 失 业 、 工 伤 和 生育 保险 的 参 保 人 员 ， 和 符合 参加 城镇 居民 基本 医疗 保险 条 件 
的 居民 。 社 会 保障 卡 可 用 于 查询 个 人 基本 信息 和 养老 、 医 疗 保险 个 人 账户 ， 办 理 住院 和 结 
算 手 续 ， 进 行医 疗 保险 个 人 账户 消费 。 

社保 卡 由 参 保 人 员 使 用 ， 在 各 个 医疗 机 构 进行 消费 ， 是 一 种 银行 卡 ， 所 有 社保 卡 的 消 
费 信息 都 在 银行 数据 库 中 有 记录 。 银 行 每 隔 一 定时 间 (通常 是 一 个 月 ) 向 A 市 人 力 资源 和 
社会 保障 局 提供 一 个 社保 卡 消费 明细 (dbf 格式)，A 市 人 力 资源 和 社会 保障 局 根据 银行 提 
供 的 数据 与 社保 管理 系统 数据 库 中 的 数据 进行 对 比 ， 并 在 核对 之 后 将 资金 结算 给 相应 的 医 
疗 机 构 。 

A 市 人 力 资源 和 社会 保障 局 现在 有 一 个 社保 管理 系统 用 于 管理 社保 相关 数据 ， 如 参 保 
人 员 信息 、 医 疗 机 构 信 息 、 社 保 机 构 信息 等 ， 社 保管 理 系统 的 服务 器 位 于 A 市 人 力 资 源 和 
社会 保障 局 ， 服 务 器 操作 系统 为 Windows， 数 据 库 管理 系统 为 Oracle。 社 保 卡 结算 系统 需 
要 运行 在 这 台 服 务 器 上 ， 并 使 用 服务 器 上 原 有 数据 ， 结 合 银行 提供 的 社保 卡 消费 明细 ， 生 
成 结算 数据 及 各 种 统计 报表 。 

银行 的 程序 与 社保 程序 是 完全 独立 的 两 个 系统 ， 完 成 不 同 的 数据 库 结 构 ， 同 一 个 医疗 
机 构 在 银行 的 数据 库 中 ， 与 社保 程序 数据 库 中 具有 完全 不 同 的 编码 ， 在 设计 开发 社保 卡 结 
算 系统 时 要 注意 数据 接口 和 编码 对 应 。 综 上 所 述 ， 社 保 卡 结算 系统 主要 需求 如 下 。 
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(1) 系统 运行 在 A 市 人 力 资源 和 社会 保障 局 现 有 服务 器 上 ， 使 用 现 有 Oracle 数据 库 。 
在 选用 开发 工具 时 ， 需 要 考虑 与 操作 系统 和 数据 库 的 兼容 性 。 

(2) 本 系统 的 用 户 有 两 类 : A 市 人 力 资源 和 社会 保障 局 人 员 ， 医 疗 机 构 人 员 。 两 类 用 
户 具有 不 同 的 权限 。 

(3) 实现 银行 数据 库 中 医疗 机 构 编 码 与 社保 管理 系统 数据 库 中 医疗 机 构 编 码 之 间 的 对 

(4) 将 银行 提供 的 dbf 格式 的 社保 卡 消费 数据 导入 到 数据 库 中 。 

(5) 对 从 银行 导入 的 数据 进行 审核 。 

(6) 对 各 个 医疗 机 构 进行 结算 ， 并 保存 结算 相关 数据 。 结 算 可 以 分 批 进行 ， 例 如 ， 应 
该 付 给 某 医疗 机 构 10 万 元 ， 可 以 分 几 次 将 款 结 清 。 

(7) 按照 社保 机 构 和 日 期 查询 社保 卡 消费 情况 ， 生 成 各 种 报表 。 


13.1.2 数据库 结 构 


社保 卡 结算 系统 涉及 三 种 类 型 的 数据 表 ， 首 先是 社保 管理 系统 中 原来 的 数据 表 ， 第 二 
是 银行 提供 的 社保 卡 消费 明细 dbf 数据 , 第 三 是 为 了 社保 卡 结算 系统 自行 设计 和 添加 的 表 。 
其 中 前 两 者 表 结 构 已 经 确定 , 程序 只 需要 使 用 即 可 。 银 行 提供 的 社保 卡 消费 明细 数据 表 (dbf 
格式 ) 包括 以 下 字段 。 
ID: 记录 编号 ，char 类 型 ， 主 键 。 
WordDate: 交易 日 期 ，date 类 型 。 
Como: 商户 号 ，char 类 型 ， 即 医院 或 者 药店 在 银行 的 客户 编号 。 
Name: 商户 名 称 ，char 类 型 ， 即 医院 或 者 药店 名 称 。 
Accno: 银行 卡号 ，char 类 型 ， 即 社保 卡号 。 
Amt: 消费 金额 ，Numeric 类 型 。 
Fee: 银联 交易 手续 费 ，Numeric 类 型 ， 按 照 消费 金额 的 一 定 比例 收取 手续 费 。 
Realamt: 实际 结算 金额 ，Numeric 类 型 ， 社 会 保障 局 将 按照 这 个 金额 与 医疗 机 构 
口 Type: 消费 类 型 ，char 类 型 ， 为 PCA 正常 、PPT 退货 、PCC 消费 冲 正 和 PPC 退货 
冲 正 这 4 个 值 之 一 。 
口 Term: 终端 机 号 ，char 类 型 ,银联 刷卡 的 终端 机 号 。 
口 Time: 消费 时 间 ，date 类 型 ， 此 笔 消 费 所 发 生 的 具体 时 间 。 
口 Csermmo: 银联 交易 流水 号 ，char 类 型 。 
在 A 市 人 力 资源 和 社会 保障 局 的 Oracle 数据 库 服务 器 上 , 与 社保 卡 消费 相关 的 有 大 约 
10 个 表 , 为 了 便于 开发 社保 卡 结算 系统 , 在 Oracle 服务 器 上 基于 多 表 连 接 创建 了 一 个 视图 ， 
名 称 为 DIS.CARD PAYOUT， 其 结构 如 下 。 
口 SBJGBH: 社保 机 构 编号 ，VARCHAR2(10) 类 型 。 
SFZHM: 身份 证 号 码 ，VARCHAR2(18) 类 型 。 
KYHZH: 银行 卡号 ，VARCHAR2(30) 类 型 。 
JYJE: 交易 金额 ，NUMBER 类 型 。 
YHJYH: 银行 交易 号 ，VARCHAR2(100) 类 型 。 


OOOOOOODO 


口 
口 
口 
口 
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口 YYBM: 医疗 机 构 编码 ，VARCHAR2(8) 类 型 。 

口 FSSJ: 交易 发 生 时 间 ，DATE 类 型 。 

在 社保 卡 结算 系统 开发 过 程 中 ， 除 了 DIS.CARD PAYOUT 视图 外 ， 还 会 用 到 以 下 四 
个 表 〈 以 下 表 结 构 中 仅 列 出 了 与 开发 本 系统 相关 的 字段 )。 

(1) 医疗 机 构 表 MD.INSTITUTION NAIL 。 

口 SBJGBH: 社保 机 构 编号 ，VARCHAR2(8) 类 型 ， 医 院 所 属 的 社保 机 构 。 

口 YYBM: 医院 编码 ，ARCHAR2(18) 类 型 。 

口 YYCM: 医院 名 称 ，VARCHAR2(50) 类 型 。 

(2) 参 保 人 员 信 息 表 MD.EMP NAITL 。 

口 GRBH: 个 人 编号 ， 即 身份 证 号 码 ，VARCHAR2(18) 类 型 。 

口 XM: 人 员 姓 名 ，VARCHAR2(20) 类 型 。 

(3) 社保 卡 账号 表 MD.CARD_ ACCOUNT。 

口 GRBH: 个 人 编号 , 即 身 份 证 号 码 , VARCHAR2(18) 类 型 , 参 保 人 员 所 属 社保 机 构 。 

口 SBJGBH: 社保 机 构 编号 ，VARCHAR2(8) 类 型 。 

口 KYHZH: 社保 卡号 ， 即 银行 卡号 ，VARCHAR2(20) 类 型 。 

(4) 社保 机 构 表 SLSI NATL。 

口 SBJGBH: 社保 机 构 编号 ，VARCHAR2(20) 类 型 。 

口 SBJGMC: 社保 机 构 名 称 ，VARCHAR2(100) 类 型 。 

为 实现 社保 卡 结算 系统 的 功能 ， 在 社保 管理 系统 数据 库 中 需要 添加 以 下 几 个 表 。 

(1) 医疗 机 构 对 应 表 MD.INSTITUTE_DICTIONARY。 

该 表 保 存 了 银行 数据 库 的 医疗 机 构 编码 与 社保 管理 系统 数据 库 医 疗 机 构 编码 之 问 的 
对 应 关系 。 在 银行 数据 库 中 商户 号 和 终端 号 对 应 某 个 医疗 机 构 ， 而 在 社保 管理 系统 中 医院 
编码 和 社保 机 构 编号 对 应 于 某 个 医疗 机 构 。 同 一 个 医疗 机 构 在 银行 数据 库 的 代码 和 社保 管 
理 系统 数据 库 的 代码 不 同 ， 为 了 实现 银行 数据 与 社保 数据 的 交流 ， 需 要 确定 这 两 个 不 同 编 
人 码 的 对 应 关系 。MD.INSTITUTE_DICTIONARY 表 保 存 了 这 个 对 应 关系 ， 表 中 各 字段 如 下 。 

口 CORNO: 商户 号 ，VARCHAR2(30) 类 型 。 

口 TERM: 终端 号 ，VARCHAR2(20) 类 型 。 

口 YYBM: 医院 编码 ，VARCHAR2(20) 类 型 。 

口 SBJGBH: 社保 机 构 编号 ，VARCHAR2(8) 类 型 。 

(2) 社保 机 构 所 属 区 县 表 MD.SBJGQX。 

该 表 描 述 了 社保 机 构 基本 信息 ， 包 括 编码 、 名 称 和 所 属 区 县 ， 主 要 用 于 按照 区 县 进行 
统计 报表 。 表 结构 如 下 。 

口 SBJGBH: 社保 机 构 编号 ，VARCHAR2(8) 类 型。 

口 SBJGMC: 社保 机 构 名 称 ，VARCHAR2(100) 类 型 。 

口 SSQX: 所 属 区 县 ，VARCHAR2(50) 类 型 。 

(3) 银行 卡 消费 明细 表 MD.BANK_TRANSACTION。 

该 表 包 含 了 从 银行 dbf 文件 导入 的 银行 卡 消费 数据 ， 表 结构 如 下 。 

口 ID: 记录 编号 ，CHAR (20) 类 型 ， 主 键 。 

口 WordDate: 交易 日 期 ，CHAR(10) 类 型 。 

口 Como: 商户 号 ，VARCHAR2(30) 类 型 ， 即 医院 或 者 药店 在 银行 的 客户 编号 。 
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DOODODOOCDD 


口 


口 
口 
口 


口 


Name: 商户 名 称 ，VARCHAR2(60) 类 型 ， 即 医院 或 者 药店 名 称 。 

Accno: 银行 卡号 ，VARCHAR2(40) 类 型 ， 即 社保 卡号 。 

Amt: 消费 金额 ，NUMBER 类 型 。 

Fee: 银联 交易 手续 费 ，NUMBER 类 型 ， 按 照 消费 金额 的 一 定 比例 收取 手续 费 。 
Realamt: 实际 结算 金额 , NUMBER 类 型 ,社会 保障 局 将 按照 这 个 金额 与 医疗 机 构 
进行 结算 。 

Type: 消费 类 型 ，VARCHAR2(10) 类 型 ， 为 4 个 值 之 一 : PCA 正常 ，PPT 退货 ， 
PCC 消费 冲 正 ，PPC 退货 冲 正 。 
Term: 终端 机 号 ，VARCHAR2(20) 类 型 ， 银 联 刷卡 的 终端 机 号 。 
Time: 消费 时 间 ，DATE 类 型 ， 此 笔 消费 所 发 生 的 具体 时 间 。 
Cserno: 银联 交易 流水 号 ，VARCHAR2(20) 类 型 。 

Audio No: 审核 编号 ，CHAR(10) 类 型 。 


(4) 银行 卡 消费 审核 表 MD.BANK_TRANSACTION_AUDIT。 
该 表 包 含 了 对 银行 卡 消费 数据 的 审核 情况 。 表 结构 如 下 。 


OoOoOoOooOooOoOoOoOoOODO DO 


AUDIT NO: 审核 编号 ，CHAR(10) 类 型 ， 主 键 。 

AUDIT_ DATE: 审核 日 期 ，DATE 类 型 ， 默 认 值 为 当前 日 期 。 
AUDIT_USER: 审核 人 ，VARCHAR2(10) 类 型 。 

YYBM: 医院 编码 ，VARCHAR2(20) 类 型 。 

SBJGBH: 社保 机 构 编 号 ，VARCHAR2(10) 类 型 。 

YYMC: 医院 名 称 ，VARCHAR2(50) 类 型 。 

AMT_SUM: 总 消费 金额 ，NUMBER 类 型 。 

FEE_ SUM: 总 手续 费 ，NUMBER 类 型 。 

REALAMT_SUM: 总 的 实际 结算 金额 ，NUMBER 类 型 。 
MONEY PAYED: 已 经 向 医疗 机 构 支付 的 金额 ，NUMBER 类 型 。 
CANCEL: 取消 审核 标志 ，CHAR(1) 类 型 ，0 表示 未 取消 ，1 表示 已 经 取消 。 


(5) 医院 权限 表 MD.USER_HOSPITAL RIGHT。 
该 表 描述 了 用 户 对 医院 数据 的 权限 信息 ， 即 某 个 用 户 是 否 有 权 查 看 某 医院 的 数据 。 表 
结构 如 下 。 


口 
口 
口 
可 | 


USER_ID: 用 户 ID，VARCHAR2(10) 类 型 。 

YYBM: 医院 编码 ，VARCHAR2(18) 类 型 。 

SBJGBH: 社保 机 构 编 码 ，VARCHAR2(8) 类 型 。 
RIGHT: 权限 ，CHAR(1) 类 型 ，1 表示 有 权 ， 和 否则 无 权 。 


(6) 数据 上 传 日 志 表 MD.UPLOAD LOG。 
该 表 记 录 了 银行 数据 上 传情 况 ， 每 当 用 户 将 银行 提供 的 dbf 格式 的 社保 卡 消费 明细 上 
传 时 ， 上 传 信息 就 会 记录 到 这 个 表 中 。 表 结构 如 下 。 


口 
口 
国 | 
口 
口 
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UPLOAD ID: 上 传 ID，NUMBER 类 型 ， 自 动 增长 ， 主 键 。 

USER ID: 上 传 用 户 ID，VARCHAR2(10) 类 型 。 

UPLOAD DATE: 上 传 日 期 ，DATE 类 型 ， 默 认 值 为 当前 日 期 。 
TOTAL RECORD: 上 传 文件 中 的 总 记录 数 ，NUMBER 类 型 。 
CORRECT RECORD: 上 传 文件 中 成 功 导入 的 记录 数 ，NUMBER 类 型 。 
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口 ERROR RECORD: 上 传 文件 中 导入 失败 的 记录 数 ，NUMBER 类 型 
数据 库 中 还 包括 权限 管理 相关 的 几 个 表 ， 表 结构 在 第 11 章 中 已 经 介绍 过 ， 此 处 不 再 
效 述 。 


13.1.3 项目 框架 


社保 卡 结算 系统 需要 运行 在 现 有 的 服务 器 上 ， 项 目 开 发 所 采用 的 技术 必须 与 服务 器 兼 
容 。 由 于 软件 开发 时 服务 器 上 仅 具 备 .NET Framework 2.0 运行 环境 ， 而 不 具备 .NET 
Framework 4.0 运行 环境 ， 且 服务 器 上 应 该 尽 可 能 少 的 安装 非 必需 的 软件 ， 所 以 本 项 目 基 
于 .NET Framework 2.0 进行 开发 。 与 框架 版 本 相对 应 ， 数 据 访问 不 能 使 用 LINQ 和 Entity 
Framework， 而 是 使 用 Enterprise Library。 

本 项 目 采 用 多 层 结构 设计 ， 使 用 实体 框架 完成 数据 访问 。 解 决 方案 中 应 该 包含 以 下 几 
个 项 目 : 业务 实体 层 、 数 据 访问 层 、 业 务 轴 辑 层 、Web 表现 层 、 单 元 测试 项 目 。 搭 建 整个 
解决 方案 框架 的 步骤 如 下 。 

(1) 创建 一 个 空白 解决 方案 AccountCheck。 

(2) 在 解决 方案 中 添加 一 个 类 库 项 目 Account.Entity 作为 业务 实体 层 。 

(3) 在 解决 方案 中 添加 一 个 类 库 项 目 AccountCheckDal 作为 数据 访问 层 。 

(4) 在 解决 方案 中 添加 一 个 类 库 项 目 AccountCheckB1 
作为 业务 逻辑 层 。 tbl 

(5) 在 解决 方案 中 添加 一 个 ASPNET Web 应 用 程序 项 Ss 

目 AccountCheckWeb 作为 表现 层 。 

(6) 在 解决 方案 中 添加 一 个 类 库 项 目 DalTest 作为 数据 
访问 层 的 单元 测试 项 目 。 

(7) 在 各 个 项 目 之 间 添 加 引用 关系 ， 表 现 层 、 业 务 逻 
辑 层 、 数 据 访问 层 都 引用 业务 实体 层 ， 业 务 罗 辑 层 引用 数 
据 访 问 层 ， 表 现 层 引用 业务 罗 辑 层 ， 测 试 项 目 引用 被 测试 图 13.1 社保 卡 结算 系统 项 目 结构 
项 目 。 整 个 解决 方案 结构 如 图 13.1 所 示 。 


13.2 ”Oracle 数据 库 简介 


人 A 市 人 力 资源 和 社会 保障 局 的 社保 管理 系统 采用 Oracle 作为 数据 库 ， 本章 所 开发 的 社 
保 卡 结算 系统 需要 使 用 此 数据 库 中 的 多 个 表 。 为 了 便于 数据 共享 和 管理 ， 本 项 目 直接 使 用 
社保 管理 系统 已 有 的 Oracle 数据 库 作为 后 台数 据 库 ， 并 根据 需要 在 其 中 创建 表 。 

对 于 大 多 数 ASPNET 程序 员 来 说 ， 所 使 用 的 后 台数 据 库 都 是 SQL Server 数据 库 ， 国 
内 的 ASPNET 开发 书籍 也 大 都 以 SQL Server 数据 库 为 例 讲解 数据 访问 功能 ， 因 此 有 必要 
在 本 项 目 正 式 开发 以 前 对 Oracle 数据 库 进 行 简单 介绍 。 


13.2.1 安装 Oracle 


Oracle 数据 库 安 装 程序 可 以 从 Oracle 官方 网 站 免费 下 载 ， 下 载 地 址 为 
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http://www.oracle.com/technology/software/index.html。Oracle 数据 库 有 多 个 版 本 ， 如 企业 版 


(Enterprise Edition)、 标 准 版 (Standard Edition )、 
版 功能 最 
较 简 单 ， 占 用 资源 少 ， 


强大 , 但 是 占用 资源 多 , 而 速成 版 功能 
且 为 免费 软件 。 速 成 版 在 


使 用 上 还 包括 其 他 一 些 限制 ， 如 仅 支 持 单 处 理 
器 ， 最 多 只 能 管理 4G 数据 和 1G 内 存 。 对 于 较 
小 规模 的 应 用 程序 来 说 , Oracle 速成 版 能 够 满足 
程序 对 数据 库 的 要 求 。 


于 本 项 目 在 开发 过 程 中 仅 用 到 最 基本 的 


Oracle 数据 库 功 能 , 所 以 笔者 在 开发 过 程 中 使 用 


的 是 


OracleXE〈 即 速成 版 )。 在 下 载 OracleXE 时 要 注意 选择 多 国语 


速成 版 (Express Edition) 等。 其 中 企业 


DO mit 2.5.3 | 

回 获取 帮助 上 
问 Rainlendar2 局 备份 数据 库 
SnagTt 9 ， 忆 还 原 数据 库 
癌 Yindows Powershell 1.0 + 时 局 动 数据 库 
问 YingAR +* 必 入门 
id ， 园 停止 数据 库 
回 阳 件 网 运行 SQL 命 信行 
加 谷歌 全 山 词霸 合作 版 2. 0 ， 网 转 至 数据 库 主 页 
加 eT » 


13.2 OracleXE 程序 菜单 
言 版 ， 这 样 才能 支持 中 


文 界面 。Oracle 的 安装 与 普通 软件 安装 类 似 , 按照 安装 向 导 的 提示 一 步 一 步 进 行 操作 即 可 。 


安装 完成 后 ，Windows 开始 菜单 中 将 出 现 以 下 内 容 ， 


13.2.2 ”管理 用 户 


如 图 13.2 所 示 。 


Oracle 数据 库 通常 需要 创建 多 个 用 户 ， 每 个 用 户 拥 有 不 同 的 数据 库 资 源 和 权限 ， 其 中 
系统 用 户 sys 具有 最 高 权限 。OracleXE 提供 一 个 Web 方式 的 图 形 化 界面 管理 工具 , 这 一 点 
与 SQL Server 有 很 大 区 别 。 从 Windows 开始 菜单 中 选择 Oracle Database 10g Express Edition| 


“转型 


| 数据 库 首页 ” 


命令 ， 则 会 打开 如 图 13.3 所 示 的 数据 库 登 录 界 面 。 


在 其 中 输入 用 户 名 


和 密码 (其 中 密码 为 安装 OracleXE 时 设置 的 密码 )， 然 后 单 击 “登录 ”按钮 。 


pplication 
文件 中 Er ET Te ET IRD 帮助 QD) 


WB c HE 


ss enterprise 
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间 击 刘 处 以 工艺 如 阿 入 站 


图 13.3 Oracle 管理 程序 登录 界面 


登录 Oracle 成 功 后 ， 


会 跳 转 到 如 图 13.4 所 示 的 Oracle 数据 库 管理 主页 ， 


其 中 显示 了 


的 管理 工具 按钮 ， 在 页 面 右 侧 显示 了 帮助 链接 和 数据 库 性 能 监视 器 。 


在 Oracle 数据 库 管理 主页 ， 依 次 选择 “管理 ” |“ 数据 库 用 户 ”| 
进入 创建 数据 库 用 户 页 面 ， 如 图 13.5 所 示 。 在 其 中 可 以 输入 要 创建 的 用 户 名 和 密码 ， 并 


户 分 配 适当 的 权限 。 


“创建 用 户 ” 命 令 ， 
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图 13.5 创建 数据 库 用户 


在 Oracle 中 创建 一 个 用 户 后 ， 会 自动 创建 一 个 与 用 户 名 相同 的 方案 (Schema)。 简 单 
来 说 ， 方 案 就 是 一 个 数据 库 对 象 〈 如 表 、 视 图 等 ) 的 容器 ， 一 个 数据 库 对 象 〈 如 表 ) 完整 
名 称 由 方案 名 与 对 象 名 〈 如 表 名 ) 组 成 ， 如 在 前 面 内 容 中 提 到 的 MD.CARD ACCOUNT 
表示 MD 方案 下 的 CARD_ACCOUNT 表 。 

在 社保 卡 结算 系统 中 ， 需 要 用 到 3 个 数据 库 用 户 和 方案 ， 分 别 是 MD、DIS 和 SI。 在 
Oracle 数据 库 管理 工具 中 添加 这 3 个 用 户 ， 并 为 其 赋予 DBA 的 用 户 权限 。 


13.2.3 ”管理 表 和 数据 


在 Oracle 中 添加 表 、 修 改 表 结 构 等 操作 与 SQL Server 不 同 ， 下 面 将 介绍 如 何在 Oracle 
中 对 表 进 行 管理 。 在 Oracle 数据 库 管 理 主页 中 ， 依 次 选择 “对 象 浏览 器 ”| “浏览 ”|“ 表 ” 
命令 ， 则 进入 对 象 浏览 器 页 面 。 页 面 左 侧 列 出 了 当前 用 户 的 所 有 表 ， 从 左 侧 选 择 一 个 表 ， 
则 在 右 侧 此 表 的 详细 信息 以 及 相关 的 操作 按钮 ， 如 图 13.6 所 示 。 


-a 


第 3 篇 项 目 实战 


很 提示 : 如 果 在 对 象 浏览 器 中 找 不 到 某 个 表 ,， 则 可 能 是 由 于 数据 库 登 录用 户 不 正确 。 例 如 ， 
图 13.6 所 示 为 MD 用 户 登录 后 对 象 浏览 器 页 面 ， 其 中 列 出 了 MD 下 的 所 有 表 ， 
却 不 显示 DIS、SI 等 其 他 用 户 的 表 。 
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图 13.6 ”对象 浏览 器 


在 图 13.6 所 示 的 对 象 浏览 器 页 面 中 ， 有 “添加 列 ””“ 修 改 列 ””“ 删 除 列 ””“ 删 除 ” 等 
多 个 功能 按钮 , 可 以 选择 这 些 按 钮 以 执行 相应 操作 。 如 果 要 查看 其 他 类 型 的 数据 库 对 象 (如 
存储 过 程 等 )， 则 从 对 象 浏览 器 页 面 左 侧 顶 部 的 下 拉 列 表 中 选择 相应 类 型 即 可 。 

如 果 要 创建 新 表 ， 则 从 对 象 13.6 所 示 页 面 中 单 击 “ 创 建 ” 按 钮 ， 在 接 下 来 的 选择 创建 
对 象 类 型 页 面 中 选择 创建 表 ， 则 进入 如 图 13.7 所 示 的 创建 表 页 面 ， 在 其 中 输入 表 名 、 各 个 
字段 名 称 和 类 型 。 


外 提示 : Oracle 数据 库 与 SQL 数据 库 的 数据 类 型 有 较 大 不 同 ，Oracle 的 varchar2 类 型 相 
当 于 SQL Server 的 varchar 类 型 ，Oracle 的 date 类 型 相当 SQL Server 的 datetime 
类 型 ，Oracle 的 number 类 型 既 可 以 表示 整数 也 可 以 表示 小 数 。 
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13.7 创建 新 表 
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在 图 13.7 所 示 页 面 中 填写 各 个 字段 后 ， 单 击 “ 下 一 步 ”按钮 ， 则 进入 设置 主键 页 面 ， 
如 图 13.8 所 示 。 在 Oracle 数据 库 中 为 表 设置 主键 时 ， 有 一 个 序列 填充 的 选项 。 这 个 选项 类 
似 于 SQL Server 数据 库 中 的 自动 增长 列 , 可 以 使 用 一 个 指定 的 序列 自动 为 新 插入 的 数据 设 


D 
置 主键 。 如 果 不 需 要 填充 主键 ， 则 选择 “未 填充 ”选项 。 
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图 13.8 设置 主键 
为 表 选 择 一 个 或 多 个 字段 作为 主键 后 ， 一 直 单 击 “ 下 一 步 ” 按 钮 ， 直 到 出 现 “ 完 成 ” 
按钮 。 单 击 “ 完 成 ”按钮 ， 则 进入 到 确认 页 面 ， 其 中 列 出 了 要 创建 表 的 详细 信息 ， 如 图 13.9 
成 了 表 的 创建 。 


所 示 。 在 其 中 单 击 “ 创 建 ”按钮 ， 就 最 终 完 


文件 由 编辑 中 ) 查看 轨 历史 人 @) 书签 如 ”工具 中 帮助 0 
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外 是 方案 :MD 

表 名 :LOGIN_USER 
约束 条 件 
©sa 


“PASSWORD” VARCHAR2(20), 
constraint “LOGIN USER PE” primary key (“USERID”) 六 
二 


TE a “LOGIN_USER” ( 
“USERI VARCHAR2 (10), 
ER VARCHARZ(20)， 


图 13.9 创建 表 确 认 页 面 


如 果 要 为 表 添 加 数据 , 则 在 图 13.6 所 示 的 对 象 浏览 器 中 选中 一 个 表 , 然后 单 击 “ 数 据 ” 
按钮 ， 再 单 击 “ 插 入 行 ” 按 钮 ， 则 进入 如 图 13.10 所 示 页 面 ， 在 其 中 可 以 向 表 中 添加 数据 。 
“423 。 


EEEEEREDZOEIC Cx] 
Er = 
[aa Ch /oem 0 rom ts ~ |e 万 
Ce 了 才 省 填 让 同 - 厂 安 - 宛 -天 < 芒 义 -图 - 
Fo 
Se 
ap mp 
ET EET TT 
LOGIN USER CE 
am 和 取消 | 创建 | 创建 和 他 理 另 -个 | 
二 oem_use 
Useid assr 


日 
加 
加 
EY 


图 13.10 向 表 中 添加 数据 


如 果 要 在 Oracle 中 执行 SQL 命令 ， 从 数据 库 首 页 上 选择 “SQL 命令 ”|“ 输 入 命令 ”， 
则 打开 SQL 命令 页 面 ， 如 图 13.11 所 示 。 由 于 数据 库 的 数据 量 通常 比较 大 ，Oracle 并 不 显 
示 全 部 的 查询 结果 ， 而 是 只 显示 结果 的 前 N 条 记录 ， 这 个 N 的 数值 可 以 从 图 13.11 所 示 页 
面 的 顶部 下 拉 列 表 中 进行 设置 。 


厅 8 动 交 里 未 [10 5] 


[elect * Eom sbjeax 


人 


7 ea 
7 a 
EGG 仙人 宝生 民有 
7 se 
和 和 pa 

[和 


图 13.11 输入 和 执行 SQL 命令 


13.2.4 ”PL/SQL 简介 


Oracle 数据 库 中 使 用 PL/SQL 作为 数据 查询 语言 ，PL/SQL 语言 是 对 标准 SQL 语言 
扩展 。 SQL Server 数据 库 使 用 的 SQL 语言 称 为 工 SQL 语言 , 也 是 对 标准 SQL 语言 的 扩展 。 
PL/SQL 语言 最 基本 的 增删 改 查 语句 与 工 SQL 语言 相同 ， 如 下 面 的 代码 所 示 。 


INSERT INTO Web User (ID, Name, Password) VALUES ('"user10'， "zhang san', 
'123456'); 


UPDATE Web User SET Name=' 张 三 " WHERE ID="'user10'; 
SELECT * FROM Web User; 


DELETE FROM Web _ User WHERE ID="'userl10'; 


PL/SQL 与 工 SQL 有 一 定 差别 ， 对 于 只 熟悉 SQL Server 数据 库 的 开发 人 员 来 说 ， 刚 开 
始 使 用 Oracle 数据 库 可 能 会 感到 不 习惯 。 下 面 主 要 介绍 几 点 PL/SQL 与 工 SQL 不 同 之 处 。 
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1. 查询 前 N 条 记录 
在 工 SQL 中 ， 使 用 TOP 关键 字 查 询 前 N 条 数据 ， 如 下 代码 所 示 。 


SELECT TOP 10 * FROM WEB USER 


在 PL/SQL 中 不 识别 TOP 关键 字 ， 要 想 查 询 前 N 条 数据 ， 需 要 使 用 ROWNUM。 


ROWNUM 表示 一 条 记录 在 查询 结果 集中 的 行 号 ， 从 1 开始 。 如 果 要 查询 前 N 条 记录 ， 则 


应 使 


即使 


以 在 
的 数 


j 以 下 语句 。 

SELECT * FROM Web User WHERE ROWNUM<=10 

需要 注意 的 是 ， ROWNUM 不 支持 大 于 比较 。 例 如 ， 下 面 的 语句 查询 结果 永远 为 空 ， 
表 中 有 多 条 数据 。 

SELECT * FROM Web User WHERE ROWNUM>1 

如 果 确 实 要 查询 结果 集中 某 个 行 以 后 的 数据 ， 则 应 该 使 用 子 查询 ， 如 下 代码 所 示 。 
SELECT * FROM 


(SELECT ID,NAME,PASSWORD, ROWNUM AS RN FROM Web User ) 
WHERE RN>1 


2. 数据 分 页 

由 于 Oracle 数据 库 的 ROWNUM 函数 能 够 标识 一 条 记录 在 整个 查询 结果 中 的 行 号 , 所 
Oracle 数据 库 中 实现 数据 分 页 功能 较为 方便 , 只 需 查 询 ROWNUM 值 在 某 个 范围 之 内 
据 即 可 。 由 于 ROWNUM 函数 不 支持 大 于 比较 , 所 以 分 页 查询 时 同样 需要 使 用 子 查询 。 


下 面 的 代码 查询 出 Web_User 表 中 的 第 2 页 数据 (每 页 10 条 数据 )。 


SELECT * FROM 
(SELECT ID,NAME,PASSWORD, ROWNUM AS RN FROM Web User ) 
WHERE RN>20 and RN<=30 


3. 使 用 变量 
在 PL/SQL 中 定义 和 使 用 变量 的 语法 与 工 SQL 也 不 相同 , 给 变量 赋值 时 要 使 用 := 符号 ， 


在 查询 语句 中 使 用 变量 时 要 在 变量 名 前 面 加 :前 缀 。 以 下 代码 是 一 个 定义 和 使 用 变量 的 


例子 


DECLARE 一 -必须 有 此 关键 字 ， 表 示 定义 变量 
var id varchar2 (10); 一 -定义 变量 时 变量 名 前 面 没 有 冒号 
Var pass varchar2(20); 

BEGIN 

var id:='admin'7 一 -给 变量 赋值 时 变量 名 前 面 没有 冒号 
Var pass:="123456"; -- 赋 值 号 为 := 


=-- 在 查询 中 使 用 变量 时 变量 名 前 面 有 冒号 
UPDATE web user SET password=:var pass WHERE id=:var id; 
END; 


仿 提 示 : 在 SQL Server 的 T-SQL 中 ， 变 量 名 以 电子 邮箱 符号 @ 为 前 级， 如 @myvar， 而 在 


Oracle 的 PL/SQL 中 ， 变 量 名 以 冒号 :为 前 级， 如 :myvar。 如 果 在 PL/SQL 中 把 变 
量 名 写成 了 以 @ 开 头 (如 @myvar) ， 命 令 执 行 时 就 会 出 现 错误 。 
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4. 执行 多 条 语句 


在 PL/SQL 中 ， 如 果 要 一 次 执行 多 条 语句 ， 那 么 这 些 语句 必须 包含 在 一 对 BEGIN 和 
END 之 内 ， 和 否则 就 会 出 现 语法 错误 。 如 下 例 所 示 。 


-- 下 面 这 两 条 语句 如 果 一 起 执行 就 会 出 现 错误 

UPDATE web user SET password='123456' WHERE id="'admin'; 
UPDATE web user SET password="'654321' WHERE id="user01'7 
-- 下 面 这 两 条 语句 可 以 一 起 执行 ， 不 会 出 错 

BEGIN 

UPDATE web user SET password='123456' WHERE id="'admin'; 
UPDATE web user SET password="'654321' WHERE id='user01'7 
END; 


5. 类 型 转换 


在 执行 数据 库 查 询 时 ， 经 常 需 要 进行 类 型 转换 。 限 于 篇 幅 ， 此 处 仅 介 绍 PL/SQL 中 常 
用 的 类 型 转换 函数 。TO_CHAR0O 函 数 实现 将 数据 以 特定 格式 转换 成 字符 类 型 ， 
TO_NUMBERO 函 数 将 数据 转换 为 数值 类 型 ，TO_DATEO 函 数 将 数据 转换 为 日 期 类 型 。 各 
函数 语法 如 下 。 

TO_CHAR (要 转换 的 数据 ,格式 字符 串 ) 


TO_NUMBER (要 转换 的 数据 ,格式 字符 串 ) 
TO_DATE (要 转换 的 数据 ,格式 字符 串 ) 


下 面 是 几 个 类 型 转换 的 例子 。 


SELECT TO CHAR(SYSDATE, 'YYYY 年 MM 月 DDH') AS today FROM DUAL; 

-- 上 一 条 语句 输出 结果 为 : 2010 年 04 月 25 日 

SELECT TO CHAR(SYSDATE, 'DS') RS today FROM DUAL; 

-- 上 一 条 语句 输出 结果 为 : 2010-04-25 

SELECT TO CHAR(SYSDATE, 'HH:MI:SS') AS NOW12, TO CHAR (SYSDATE, 'HH24:MI:SS') 
AS NOW24 FROM DUAL; 

-- 上 一 条 语句 输出 结果 为 : 03:38:18 15:38:18 

SELECT TO CHAR(12345.67,'99,999.99') RS num FROM dual 

-- 上 一 条 语句 输出 结果 为 : 12,345. 67 


SELECT TO DATE('2010-01-01', "YYYY-MM-DD') AS my date FROM DUAL; -- 字 符 
转换 为 日 期 

SELECT TO NUMBER('123456.78') RS my number FROM DUAL; 一 字符 转换 
为 数值 


13.3 ” 母 版 页 设计 


社保 卡 结算 系统 中 大 部 分 页 面 布局 都 包含 三 部 分 : 页 头 、 正 文 和 页 脚 ， 其 中 页 头 显 示 
工具 栏 ， 页 脚 显 示 网 站 说 明和 联系 方式 ， 这 两 部 分 内 容 对 于 大 多 数 页 面 来 说 是 相同 的 ， 可 
以 放 到 母 版 页 中 。 本 节 将 介绍 母 版 页 的 设计 和 实现 。 
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13.3.1 Header 用 户 控件 


Header 用 户 控件 位 于 页 面 顶端 , 用 于 显示 程序 名 称 、 当 前 日 期 、 当 前 用 户 等 提示 信息 ， 


并 以 图 形 按钮 方式 显示 功能 导航 栏 。Header 用 户 控件 外 观 如 图 13.12 所 示 。 
_A_ 市 劳动 局 社保 卡 结 算 寺 账 系统 


医保 账目 村 对 银行 数据 上 传 可 医疗 机 构 字典 用 户 权限 管理 


图 13.12 Header 用 户 控件 


(1) 在 表现 层 项 目 AccountCheckWeb 中 添加 一 个 文件 夹 UserControls， 此 文件 夹 用 


存放 项 目 中 的 用 户 控件 。 


于 


(2) 在 UserControls 文件 夹 中 添加 一 个 用 户 控 件 Header.ascx， 在 Header.ascx 文件 中 添 


加 以 下 代码 。 


<%@ Control Language="C#" AutoEventWireup="true" CodeBehind="Header. 
ascx.cs" Inherits= "AccountCheckWeb.UserControls.Header" %> 
<div style="background:#f2f2fe; width:1000px; text-align:left;"> 
<div> 
<img src="../images/title.png" alt="" /> <!-- LOGO --> 
<span style="margin-left:auto; margin-right:30px;"> 
<asp:Label ID="Labell" runat="server" Text=""></asp:Label> 

<!-- 显示 用 户 名 称 --> 
当前 日 期 ; <s=DateTime.Now.ToLongDateString() +"gnbsp;" +tDateTime.Today. 
DayOfWeek $> </span> 
<span style="margin-left:30px;"> 


<asp:LinkButton ID="LinkButtonl" runat="server" onclick="LinkButtonl 


GEiekw> 

[注销 登录 ] </asp:LinkButton></span> 

</div> 

<! 一 以 下 内 容 显 示 各 个 导航 菜单 --> 

<div id="NavMenu"> 

<div class="ImageMenu" runat="server" id="check"> 

<img src=". ./images/check.png" alt=" 医 保 账目 核对 " /><br/> 

<a href="../Default/CheckAccount2.aspx"> 医 保 账目 核对 </a></div> 
<div class="ImageMenu" runat="server" id="upload"> 

<img src=". ./images/upload.png" alt=" 银 行 数据 上 传 ” /><br/> 

<a href="../Default/UploadData.aspx"> 银 行 数据 上 传 </a></div> 

<div class="ImageMenu" runat="server" id="Div1"> 

<img src="../images/calculator.png”alt=" 数 据 查询 统计 " /><br/> 

<a href="../Report/ConsumeReportPage.aspx"> 数 据 查询 统计 </a></div> 
<div class="ImageMenu" runat="server" id="Div2"> 

<img src="../images/floppy.png”alt=" 费 用 二 次 结算 " /><br/> 

<a href="../Default/PayAgainPage.aspx"> 费 用 二 次 结算 </a></div> 
<div class="ImageMenu" runat="server" id="dictionary"> 

<img src="../images/dictionary.png"” alt=" 定 点 医疗 机 构 字 典 " /><br/> 
<a href="../Default/InstituteDictionary.aspx"> 医 疗 机 构 字 典 </a></div> 
<div class="ImageMenu" runat="server" id="user"> 

<img src="../images/users.png”alt=" 用 户 权 限 管理 " width="48" /><br/> 
<a href="../Default/ManageUser.aspx"> 用 户 权限 管理 </a></div> 

</div> 

</div> 
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(3) 在 Header 用 户 控 件 的 Page_ Load 事件 中 ， 显 示 当 前 登录 的 用 户 名 称 ， 并 根据 用 户 
类 型 〈 医 院 用户 和 社保 机 构 用 户 ) 设置 导航 菜单 的 可 见 性 。 医 院 用 户 不 允许 使 用 “用 户 字 
典 ”“ 权 限 管理 ”“ 数 据 上 传 ”功能 ， 将 这 些 按钮 设置 为 不 可 见 。 


protected void Page Load(object sender, EventArgs e) 
lL 


WebUser user0 = Common.WebUtility.currentUser; // 得 到 当前 登录 用 户 
bool isHospital = WebUtility.isMedicineShopUser(); // 是 否 医院 用 户 
Label1.Text = "当前 用 户 : " + user0.namet+"&gnbsp; &nbsp;"; 
/* 医院 用 户 与 非 医院 用 户 ( 即 社保 机 构 用 户 ) 具 有 不 同 权限 
* 如 果 是 医院 用 户 ， 则 禁用 用 户 字典 功能 、 用 户 管理 功能 、 数 据 上 传 功能 */ 
if (isHospital) 
{ 
dictionary.Visible = false; 
user.Visible = false; 
upload.Visible = false; 
} 
else 
{ 
dictionary.Visible = true; 
user.Visible = true; 
upload.Visible = true; 


有 


(4) 在 “注销 登录 ”按钮 的 Click 事件 中 ， 注 销 当 前 用 户 ， 并 跳 转 到 登录 页 面 。 
protected void LinkButton1 Click(object sender, EventArgs e) 
1 

Common .WebUtility.currentUser = null; // 清 除 Session 中 保存 的 用 户 


Response.Redirect ("../login.aspx"); // 跳 转 到 登录 页 面 
} 


13.3.2 ”Footer 用 户 控 件 


Footer 用 户 控件 用 于 显示 用 户 单位 ( 即 A 市 人 力 资源 与 社会 保障 局 ) 地 址 、 联 系 电话 
和 版 权 信 息 。Footer 用 户 控 件 代码 如 下 : 


<%@ Control Language="C#" AutoEventWireup="true" CodeBehind="Footer. 
ascx.cs" Inherits="AccountCheckWeb.UserControls.Footer" 当 > 
<div style="width:1000px; height:20px; font-size:l2px; color:#333; 
text-align:center; 

border:dashed lpx silver; background:#f2f2f2; float:none; clear:both;"> 
主办 单位 : A 市 劳动 和 社会 保障 局 。 地 址 :A 市 府 右 街 37 号 。 联 系 电话 :12333( 传 
真 ) 0543-3162837。 
<br /> 技术 支持 : <a href="mailto:sun.j.1.studio@gmail.com"> sun.j.1.studio- 
@gmail. com </a></div> 


13.3.3 ”和 母 版 页 


社保 卡 结算 系统 母 版 页 主要 包括 3 部 分 内 容 : 顶部 的 Header 控件 、 底 部 的 Footer 控 
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件 和 页 面 中 间 的 内 容 占 位 符 控件 。 创 建 母 版 页 的 步骤 如 下 。 
(1) 在 Default 文件 夹 中 添加 一 个 母 版 页 Nrcm.master。 
(2) 拖 动 一 个 Headerascx 用 户 控件 放 在 母 版 页 顶部 。 
(3) 拖 动 一 个 Footerascx 用 户 控件 放 在 母 版 页 底部 。 
(4) 在 母 版 页 中 定义 浏览 器 标题 ， 并 导入 CSS。Nrcm.master 代码 如 下 : 


<head runat="server"> 
<title>A 市 劳动 局 社保 卡 结算 对 账 系 统 </title> 
<link href="../css/NrcmCommon.css" type="text/css" rel="Stylesheet" /> 
<asp:ContentPlaceHolder ID="head" runat="server"> 
<!-- 页 面 头 部 内 容 占 位 符 --> 
</asp:ContentPlaceHolder> 
</head> 
<body><div id="main"> 
<form id="forml" runat="server"> 
<div> 
<ucl:Header ID="Headerl" runat="server" /> 
</div> 
<div> 
<asp:ContentPlaceHolder ID="ContentPlaceHolder1"” runat="server"> 
<!-- 页 面 中 间 内 容 占 位 符 --> 
</asp:ContentPlaceHolder> 
</div> 
<div> 
<uc2:Footer ID="Footerl" runat="server" /> 
</div> 
</form> 
</div> 
</body> 


母 版 页 外 观 如 图 13.13 所 示 。 


医疗 机 构 字典 用 户 权限 管理 
主 六 单位 >。 币 芝 动 和 社会 保 随 局 -于 让 :而 皇 硬 省 号 -联系 电话 -12555 
技术 支持 : sun j 1 _ studioepmail eon 
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图 13.13 社保 卡 结算 系统 母 版 页 


13.4 权限 管理 


社保 卡 结算 系统 的 权限 管理 模块 主要 功能 和 设计 思路 与 第 11 章 所 介绍 的 基本 相同 , 但 
是 在 此 基础 上 ， 又 有 一 些 个 性 化 的 要 求 。 这 里 本 系统 使 用 企业 库 (Enterprise Library) 实现 
数据 访问 功能 ， 而 在 第 11 章 中 使 用 实体 框架 (Entity Framework) 实现 数据 访问 功能 ， 所 
以 二 者 的 实体 类 和 数据 访问 层 代 码 有 较 大 不 同 。 
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13.4.1 用 户 和 权限 管理 概述 


社保 卡 结算 系统 有 两 类 用 户 : 社保 机 构 和 医疗 机 构 。 社 保 机 构 是 指 A 市 人 力 资源 和 社 
会 保障 局 及 其 下 属 机 构 ， 医 疗 机 构 是 指 各 级 医院 和 药店 。 其 中 社保 机 构 的 用 户 和 权限 管理 
方式 与 第 11 章 所 介绍 的 方法 完全 相同 ， 此 处 不 再 袭 述 。 但 是 对 于 医疗 机 构 来 说 ,这 种 权限 
管理 方式 不 太 合 适 。 首 先 ， 医 疗 机 构 众 多 ， 如 果 为 每 个 医疗 机 构 的 工人 人 员 都 在 系统 中 创 
建 登录 用 户 名 ， 则 工作 量 太 大 。 另 外 ， 所 有 医疗 机 构 的 权限 完全 相同 且 固定 不 变 ， 其 权限 
仅 限 于 查询 自己 医院 〈 或 药店 ) 的 消费 数据 。 
日 于 医疗 机 构 众多 ， 而 且 医 疗 机 构 的 权限 是 固定 的 ，A 市 人 力 资源 和 社会 保障 局 的 用 
户 要 求 不 需要 给 医疗 机 构 创 建 用 户 和 角色 , 而 是 让 医疗 机 构 使 用 各 自 的 医院 编码 进行 登录 。 
各 个 医疗 机 构 的 数据 已 经 在 社保 管理 系统 数据 库 的 MD.INSTITUTION_NATL 表 中 存在 ， 
医疗 机 构 可 以 使 用 医院 编码 字段 YYBM 的 内 容 作 为 用 户 名 ， 社 保 机 构 编号 字段 SBJGBH 
的 内 容 作为 密码 进行 登录 。 根 据 上 述 需 求 ， 在 登录 界面 中 就 需要 根据 两 种 不 同 的 用 户 类 型 
进行 判断 。 

前 面 所 述 的 权限 管理 的 对 象 都 是 基于 功能 模块 (或 者 页 面 )， 在 社保 卡 结算 系统 中 还 
需要 另外 一 种 权限 管理 方式 ， 即 基于 医疗 机 构 的 权限 管理 。 具 体 来 说 ， 对 于 不 同 的 用 户 和 
角色 ,他 们 都 有 权限 访问 某 个 模块 (如 查询 医疗 机 构 消 费 明细 等 ), 但 是 各 个 用 户 所 允许 访 
问 的 医疗 机 构 是 不 同 的 , 例如 用 户 A 可 能 允许 访问 所 有 医疗 机 构 的 数据 , 用 户 B 只 允许 访 
问 其 中 5 个 医疗 机 构 的 数据 。 系 统管 理 员 可 以 指定 某 个 用 户 是 否 可 以 访问 某 医院 的 数据 。 

这 种 权限 控制 与 前 述 的 基于 页 面 访问 的 权限 控制 不 同 ， 不 同 用 户 都 可 以 访问 同一 个 页 
面 , 而 且 可 以 使 用 同一 页 面 上 所 有 控件 (如 单 击 按钮 以 执行 相应 功能 ), 但 是 他 们 看 到 的 数 
据 是 不 相同 的 。 要 实现 这 种 权限 管理 ， 必 须 使 用 一 种 不 同 于 前 述 权 限 管理 思路 的 方式 ， 具 
体 实现 见 本 节 后 续 内 容 。 


13.4.2 ”数据 访问 辅助 类 


社保 卡 结算 系统 使 用 Enterprise Library 实现 数据 访问 功能 , Enterprise Library 中 的 类 如 
Database 等 提供 了 大 多 数 本 项 目 所 需要 的 功能 。 项 目 中 还 包含 一 个 自 定义 的 数据 访问 辅助 
类 MyUtility， 其 中 封装 了 一 些 常 用 的 数据 访问 操作 。 


// 通 用 数据 访问 类 
internal static class MyDbUtility 
internal static Database oracle = DatabaseFactory.CreateDatabase 
("oracle") ;// 数 据 库 对 象 
/// <summary> 
/// 得 到 查询 语句 返回 的 记录 总 数 
/// </summary> 
/// <param name="sql"> 查 询 命 令 文 本 </param> 
/// <returns> 记 录 总 数 </returns> 
internal static int getCount (string sql) 
上 
Database oracle = DatabaseFactory.CreateDatabase ("oracle"); 
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} 

Wy 
/// 
VE 
fd 
Ch 


string te "gelect count(*) from (™ + ql + ™)”s; 
DbCommand command = oracle.GetSqlSstringCommand( temp); 
object o = oracle.ExecuteScalar (command); 
command.Dispose(); 

return Convert.ToInt32 (0); 


<summary> 

得 到 查询 命令 返回 的 记录 总 数 

</summary> 

<param name="command"> 数 据 库 命令 </param> 
<returns> 记 录 总 数 </returns> 


internal static int getCount (DbCommand command) 


{ 


} 
pf 


HN 
WA 
Wh 
Wh 


return getCount (command.CommandText); 


<summary> 

检查 表 是 否 在 数据 库 中 存在 

</summary> 

<param name="name"> 要 检查 的 表 名 </param> 
<returns> 是 否 存 在 </returns> 


internal static bool checkTableExists(string name) 


{ 


上 

DOA 
RA 
Wi 
BA 
Wh 


string check = "select count (*) from user tables where table name= 
upper ('"tnamet"')"; 

Database oracle = DatabaseFactory.CreateDatabase ("oracle"); 
DbCommand command = oracle.GetSqlStringCommand (check); 

return Convert.ToInt32 (oracle.ExecuteScalar (command)) > 0; 


<summary> 

删除 指定 表 (如果 表 存 在 》 

</summary> 

<param name="name"> 表 名 </param> 
<returns> 删 除 表 返回 1， 否 则 返回 0</returns> 


internal static int dropTableIfExists(string name) 


{ 


if(checkTableExists (name)) 

{ 
Database oracle = DatabaseFactory.CreateDatabase ("oracle"); 
oracle.ExecuteNonQuery (CommandType.Text, "drop table "+name); 
return 1; 

. 


return 0; 


为 了 尽 可 能 提高 MyDbUtility 类 的 正确 性 ， 及 时 发 现存 在 的 错误 ， 应 编写 一 个 单元 测 
试 类 对 MyDbUtility 类 进行 测试 ， 测 试 类 代码 如 下 : 


[TestClass()] 
public class MyDbUtilityTest 


{ 


[TestMethod () ] 


U 


public void checkTableExistsTest() 


string name = "BANK TRANSACTION"; // 存 在 的 表 名 
bool actual; 
actual = MyDbUtility.checkTableExists (name); 
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Assert.AreEqual (true, actual); 

name="NoSuchTable"; // 不 存在 的 表 名 
actual=MyDbUtility.checkTableExists (name); 
Assert.AreEqual (false, actual); 


[TestMethod () ] 
public void getCountTest () 


1 
string sql = " select * from md.Web User where id like "userg'"7 
// 查 找 多 条 记录 
int actual; 
actual = MyDbUtility.getCount (sql); 
Assert.AreEqual (3, actual); 
sql = "select * from md.Web User where id="'admin'"; 
// 查 找 一 条 记录 
actual = MyDbUtility.getCount (sql); 
Assert.AreEqual (1, actual); 
sql = "select * from md.Web User where id='nosuchuser'"; 
// 查 找 不 存在 记录 
actual = MyDbUtility.getCount (sql); 
Assert.AreEqual (0, actual); 
} 
} 
13.4.3 角色 管理 


角色 管理 模块 可 以 浏览 、 添 加 、 删 除 、 修 改 角色 数据 ， 但 是 系统 管理 员 角 色 不 允许 被 
删除 和 修改 。 实 现 角色 管理 模块 的 具体 步骤 如 下 。 

(1) 在 实体 类 项 目 Account Entity 中 添加 一 个 类 UserRole， 以 描述 角色 信息 。 

public class UserRole 


, 
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public const string RdminRoleId = "01"; // 系 统管 理 员 角色 为 01， 不 可 删除 
Private const string HospitalRoleId = "hospital"; // 医 院 角色 编号 
public string id { get; set; } 
public string name { get; set; } 
public string description { get; set; } 
#region 医院 角色 
private static UserRole hospitalRole=new UserRole() { id = Hospital 
RoleId }; 
public static UserRole HospitalRole 
{ 
get 
{ 


} 


return hospitalRole; 


} 


#endregion 


/ /判断 角色 是 否 系统 管理 员 
public bool isAdmin() 
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| 


return id == AdminRoleId; 


(2) 在 数据 访问 层 项 目 AccountCheckDAL 中 添加 一 个 类 RoleDB， 以 实现 与 角色 相关 
的 数据 访问 功能 。 
// 用 户 角 色 数 据 访问 类 


public 
| 


static class RoleDB 


public static List<UserRole> getAll () // 得 到 所 有 角色 数据 


上 

Wd 
WA 
A 
WO 
PA 
pub 
i 


} 

/// 
WA 
WA 
AA 


A 
pub. 
{ 


string sql = "select ID,NAME,DESCRIPTION FROM User Role"; 
var oracle = MyDbUtility.oracle; 
Var reader = oracle.ExecuteReader (System.Data.CommandType.Text, 
5q1); 
var list = new List<UserRole>(); 
while (reader.Read()) 
{ 
list.Add (fromDataReader (reader)); 
} 
reader.Close(); 
return list; 


<summary> 

根据 角色 ID 得 到 角色 数据 

</summary> 

<param name="id"> 要 查询 的 角色 ID</param> 
<returns> 查 询 到 的 角色 数据 </returns> 
lic static UserRole getByID (string id) 


string sql = "select ID,NAME,DESCRIPTION FROM User Role where 
ID=:id"; 

var oracle = MyDbUtility.oracle; 

DbCommand command = oracle.GetSqlStringCommand(sql); 
oracle.AddInParameter (command, ":id", DbType.string, id); 

Var reader=oracle.ExecuteReader (command); 

UserRole result = null; 

if (reader.Read()) 

{ 


result = fromDataReader (reader); 


} 
reader.Close(); 
return result; 


<summary> 

添加 角色 数据 

</summary> 

<param name="role"> 要 添加 的 角色 </param> 


<returns> 添 加 的 行 数 </returns> 
lic static int add(UserRole role) 


string sql = "insert into User Role (ID,NAME,DESCRIPTION) values 
(:id, :name, :description)"; 

Var oracle = MyDbUtility.oracle; 

DbCommand command = oracle-GetSqlStringCommand (sql); 
oracle.AddInParameter (command, ":id", DbType.Sstring, role.id); 
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(3) 在 数据 访问 层 测 试 项 目 DalTest 中 , 为 RoleDB 添加 一 个 单元 测试 类 并 运行 ， 以 检 


oracle.AddInParameter (command, ":name", DbType.String, role. 
name); 
oracle.AddInParameter (command, ":description", DbType.string, 
role. description); 
return oracle.ExecuteNonQuery (command); 

} 

/// <summary> 

/// 更 新 角色 数据 

/// </summary> 

/// <param name="role"> 要 更 新 的 角色 </param> 

/// <returns> 被 更 新 的 行 数 </returns> 

public static int update (UserRole role) 


string sql = "update User Role set NAME=:name, DESCRIPTION=:descry- 


ption where ID=:id"; 
var oracle = MyDbUtility.oracle; 
DbCommand command = oracle.GetSqlStringCommand (sql); 
oracle.AddInParameter (command, ":id", DbType.String, role.id); 
oracle.AddInParameter (command, ":name", DbType.String, role. 
name); 
oracle.RddInParameter (command, ":description", DbType.string, 
role. description); 
return oracle.ExecuteNonQuery (command); 

} 

/// <summary> 

/// 根据 角色 ID 删除 角色 

/// </summary> 

/// <param name="id"> 角 色 ID</param> 

/// <returns> 被 删除 的 行 数 </returns> 

public static int delete (string id) 

{ 
string sql "delete from User Role where ID=:id"; 
var oracle = MyDbUtility.oracle; 
DbCommand command = oracle.GetSqlStringCommand(sql); 
oracle.AddInParameter (command, ":id", DbType.string, id); 
return oracle.ExecuteNonQuery (command); 


i 

/// <summary> 

/// 根据 数据 读 取 器 的 当前 记录 生成 一 个 UserRole 对 象 

/// </summary> 

/// <param name="reader"> 数 据 读 取 器 </param> 

/// <returns> 生 成 的 UserRole 对 象 </returns> 

private static UserRole fromDataReader (IDataReader reader) 
UserRole role = new UserRole(); 
role.description = reader["description"] .ToString(); 
role.id = reader["id"] .ToString(); 
role.name = reader["name"] .ToString(); 
return role; 


测 和 修改 RoleDB 类 的 错误 。 


[TestClass] 
public class RoleDbTest 


L 
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public void testAddRole() 
ft 
UserRole role = new UserRole() { id = "**", name = "test", description 
= null }; 
int n= RoleDB.add (role); 
Assert.AreEqual (1, n); 
role = RoleDB.getByID("**"); // 验 证 数据 库 中 已 存在 新 角色 
Assert.IsNotNull (role); 
} 
[TestMethod] 
public void testDeleteRole() 
| 
int n=RoleDB.delete("**"); 
Assert.AreEqual (1, n); 
var role = RoleDB.getByID("**"); // 数 据 库 已 不 存在 指定 角色 
Assert.IsNull (role); 


TestMethod] 
public void testGetAll () 


var list = RoleDB.getAll(); 
Assert.IsTrue(list.Count > 0); 


TestMethod] 
public void testGetByID() 


Var role = RoleDB.getByID("01"); 


Assert .AreEqual ("系统 管理 员 "， role.name); // 角 色 存 在 且 数 据 正确 
role = RoleDB.getByID ("norole"); 


Assert.IsNull (role); // 角 色 不 存在 


} 

(4) 在 业务 逻辑 层 项 目 AccountCheckBLL 中 添加 一 个 RoleBLL 类 ， 这 个 类 的 主要 功 
能 为 调用 数据 访问 层 的 相应 方法 ， 代 码 与 第 11 章 相同 ， 此 处 省 略 。 

(5) 在 表现 层 项 目 AccountCheckWeb 中 添加 一 个 页 面 UserRolePage.aspx， 代 码 与 第 
11 章 相 同 ， 此 处 省 略 。 


13.4.4 用 户 管理 


户 管 理 横 块 可 以 查询 、 浏 览 、 添 加 、 删 除 、 修 改 用 户 信息 。 实 现 用 户 管理 模块 的 具 
体 步骤 如 下 。 
(1) 在 业务 实体 项 目 Account Entity 中 添加 一 个 类 WebUser 以 描述 用 户 信息 。 


public class WebUser 
有 


public string id { get; set; } 
public string name { get; set; } 
public string password { get; set; } 


public UserRole role { get; set; } // 用 户 所 属 角色 
public bool isAdmin() // 是 否 管理 员 
{ 


return this.role.isAdmin(); 


} 
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(2) 在 数据 访问 层 项 目 AccountCheckDAL 中 添加 一 个 类 WebUserDB， 实 现 与 用 户 相 
关 的 数据 访问 功能 。 
public static class WebUserDB 
{ 
#region 私有 字段 ( 列 名 和 表 名 ) 
private const string IdColumn = "ID"; 
private const string NameColumn = "Name"7 
private const string PasswordColumn = "Password"; 
private const string RoleColumn = "Role"; 
private const string WebUserTable = "MD.Web User"; 
#endregion 
// 得 到 所 有 用 户 
public static List<WebUser> getAllUsers() 
[ 
List<WebUser> list = new List<WebUser> () 7 
Database oracle = MyDbUtility.oracle; 
// 创 建 select 语句 
String sql = string.Format ("select {0},{1},{12},{13} from {4}", 
IdCcolumn,NameCcolumn,PasswordCcolumn，RoleCcolumn，WebUser-- 
Table) 
DbCommand command = oracle.GetSqlStringCommand (sql); 
// 执 行 命令 ， 得 到 数据 读 取 器 ， 读 取 数 据 
using (IDataReader reader = oracle.ExecuteReader (command)) 
{ 
while (reader.Read()) 
. 
list.Add(fromDataReader (reader)); 
F 
} 
return list; 
/// <summary> 
/// 修改 用 户 不 修改 密码 ) 
/// </summary> 
/// <param name="user"> 要 修改 的 用 户 信息 </param> 


public static int updateUser (WebUser user) 


{ 
/ /构建 update 语句 
string sql = string.Format ("update {0} set {1}=:name, {2}=:role where 
{3}=:id", 


WebUserTable, NameColumn, RoleColumn, IdColumn); 

Database oracle = MyDbUtility.oracle; 
/ /创建 命令 ， 添 加 参数 ， 并 执行 命令 
DbCommand command = oracle.GetSqlstringCommand (sql); 
oracle.AddInParameter (command, ":name", DbType.Sstring, user. 
name); 
oracle.AddInParameter (command, ":id", DbType.String, user.id); 
oracle.AddInParameter (command, ":role", DbType.String, user.role. 
id); 
return oracle.ExecuteNonQuery (command); 

b 

/// <summary> 

/// 修改 密码 

/// </summary> 

/// <param name="id"> 用 户 ID</param> 

/// <param name="pass"> 新 密码 </param> 

public static int changePassword(string id, string pass) 
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} 


// 构 建 update 语句 

string sql = string.Format ("update {0} set {1}=:pass where {2}=:id", 
WebUserTable, PasswordColumn, IdColumn); 

Database oracle = MyDbUtility.oracle; 

// 创 建 命令 ， 添 加 参数 ， 并 执行 命令 

DbCommand command = oracle.GetSqlStringCommand (sql); 
oracle.AddInParameter (command, ":pass", DbType.String, pass); 
oracle.AddInParameter (command, ":id", DbType.Sstring, id); 

return oracle.ExecuteNonQuery (command); 


/// <summary> 

/// 根据 用 户 ID 得 到 用 户 信息 

/// </summary> 

/// <param name="id"> 用 户 ID</param> 

public static WebUser getUserByID(string id) 


Database oracle = MyDbUtility.oracle; 
// 构 建 select 语句 
string sql = string.Format ("select {0},1{1},1{2},1{3} from {4} where 
{0}=:id", 
IdColumn, NameColumn, PasswordColumn, RoleColumn, WebUser-— 
Table); 
DbCommand command = oracle.GetSqlStringCommand (sql); 
oracle.RddInParameter (command, ":id", DbType.Sstring, id); 
WebUser user = null; 
/ /执行 查询 ， 得 到 数据 读 取 器 ， 读 取 数 据 
using (IDataReader reader = oracle.ExecuteReader (command)) 
{ 
if (reader.Read()) 


D 


user = fromDataReader (reader); 


ih 


return user; 


/// <summary> 

/// 根据 DataReader 当前 记录 得 到 一 个 用 户 对 象 

/// </summary> 

/// <param name="reader">DataReader 对 象 </param> 

private static WebUser fromDataReader (IDataReader reader) 


{ 


} 


WebUser user = new WebUser () 
{ 
// 根 据 DataReader 各 个 字段 设置 WebUser 各 个 属性 
id = reader["id"] .ToString() .Trim()， 
name = reader["name"] .ToString()， 
password = reader["password"] .ToString () 
}; 
string temp = reader[3] .ToString(); 
user.role = RoleDB.getByID (temp); 
return USer; 


/// <summary> 


/// 删除 用 户 


/// </summary> 


/// <param name="id"> 要 删除 的 用 户 ID</param> 
public static int deleteUser (string id) 


{ 
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string sql = string.Format ("delete from {0} where {1}=:id", 
WebUserTable, IdColumn); 
Database oracle = MyDbUtility.oracle; 
DbCommand command = oracle-GetSqlStringCommand (sql); 
oracle.AddInParameter (command, ":id", DbType.Sstring, id); 
return oracle.ExecuteNonQuery (command); 

} 

/// <summary> 

/// 添加 用 户 〈 新 添加 的 用 户 密码 默认 为 123456) 

/// </summary> 

/// <param name="user"> 要 添加 的 用 户 信 息 </param> 

public static int addUser (WebUser user) 
//return addUser (user.ID，user.Name，user-Password) 7 
string sql = string.Format ("insert into {0} ({1},1{2},1{3},1{4}) values 
(:id, :name, '123456', :role)", 

WebUserTable, IdColumn, NameColumn, PasswordColumn, RoleCo- 
lumn); 

Database oracle = MyDbUtility.oracle; 
DbCommand command = oracle.GetSqlStringCommand(sql); 
oracle.RddInParameter (command, ":name", DbType.String, user. 
name); 
oracle.RddInParameter (command, ":id", DbType.String, user.id); 
oracle.RddInParameter (command, ":role", DbType.String, user.role. 
id); 
return oracle.ExecuteNonQuery (command); 


} 


(3) 在 数据 访问 层 测 试 项 目 DalTest 中 ， 为 WebUserDB 添加 一 个 单元 测试 类 并 运行 。 
测试 类 代码 如 下 : 


[TestClass] 

public class UserDBTest 

{ 

TestMethod] 

public void testGetUserByID() // 测 试 根据 ID 得 到 用 户 


string id = "admin"; 

WebUser user = WebUserDB.getUserByID (id); 
Assert.IsNotNull (user); 

Assert.AreEqual (id, user.id); 


TestMethod] 
public void testUpdateUser () // 测 试 修改 用 户 


string id = "admin"; 
WebUser user = WebUserDB.getUserByID (id); 
user.name = "管理 员 0"; 
int n = WebUserDB.updateUser (user); 
Assert.AreEqual (1, n); 
user = WebUserDB.getUserByID (id); 
Assert.AreEqual ("admin", user.id); 
Assert.AreEqual ("管理 员 0", user.name); 
} 
[TestMethod] 
public void testaddUser () // 测 试 添加 用 户 
UserRole role = new UserRole() { id = "01" }; 
WebUser user = new WebUser() { id = "new01", name = "new user", role 
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= role }; 
int n = WebUserDB.addUser (user); 
Assert.AreEqual (1, n); 
Var user2 = WebUserDB.getUserByID ("new01"); 
Assert.IsNotNull (user2); 
Assert.IsTrue (user.password.Length > 0); 
} 
[TestMethod] 
public void testDeleteUser () // 测 试 删除 用 户 
{ 
string id = "new01"; 
int n=WebUserDB.deleteUser (id); 
Assert.AreEqual (1, n); 
var user = WebUserDB.getUserByID(id); 
Assert.IsNull (user); 


} 

(4) 添加 业务 轴 辑 层 类 WebUserBLL， 代 码 与 第 11 章 相 同 ， 此 处 省 略 。 

(5) 在 表现 层 项 目 AccountCheckWeb 中 添加 一 个 页 面 ManagerUseraspx， 页 面 代码 与 
第 11 章 相 同 ， 此 处 省 略 。 


13.4.5 功能 模块 管理 


功能 模块 管理 可 以 对 系统 中 的 功能 模块 进行 浏览 、 添 加 、 删 除 、 修 改 操作 。 实 现 功能 
模块 管理 的 具体 步骤 如 下 。 

(1) 在 业务 实体 项 目 Account. Entity 中 添加 一 个 类 ApplicatonModule， 以 描述 功能 模块 
信息 。 


public class ApplicationModule 
{ 


public string id { get; set; } / /模块 编号 
public string name { get; set; } // 模 块 名 称 
public string description { get; set; } // 模 块 描述 
public string url { get; set; } // 模 块 对 应 页 面 的 URL 


. 
(2) 在 数据 访问 层 项 目 AccountCheckDAL 中 添加 一 个 类 ApplicationModuleDB， 以 实 
现 与 功能 模块 相关 的 数据 访问 功能 。 
public static class ApplicationModuleDB 
{ 
// 得 到 所 有 功能 模块 列表 
public static List<ApplicationModule> getAll]l () 
{ 
string sql = "select ID, NAME, URL, DESCRIPTION from Applicati-— 
on Module"; 
var oracle 
Var reader 
sql); 
Var list = new List<ApplicationModule>(); 
while (reader.Read()) 
{ 


MyDbUtility.oracle; 
oracle.ExecuteReader (System.Data.CommandType.Text, 


[el 


list.Add (fromDataReader (reader)); 
} 


reader.Close(); 
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} 


return list; 


// 根 据 模块 编号 得 到 模块 对 象 


public static ApplicationModule getByID(string id) 


{ 


} 


string sql = "select ID, NAME, URL, DESCRIPTION from Application 
Module where ID=:id"; 
var oracle = MyDbUtility.oracle; 
DbCommand command = oracle.GetSqlStringCommand(sql); 
oracle.AddInParameter (command, ":id", DbType.Sstring, id); 
Var reader = oracle.ExecuteReader (command); 
ApplicationModule result = null; 
if (reader.Read()) 
{ 

result = fromDataReader (reader); 
} 
reader.Close(); 
return result; 


// 根 据 页 面 url 得 到 模块 对 象 
public static ApplicationModule getByUr]l (string url) 


string sql = "select ID, NAME, URL, DESCRIPTION from Application 
Module where url=:url"; 
var oracle = MyDbUtility.oracle; 
DbCommand command = oracle.GetSqlStringCommand(sql); 
oracle.AddInParameter (command, ":url", DbType.Sstring, url); 
Var reader = oracle.ExecuteReader (command); 
ApplicationModule result = null; 
if (reader.Read()) 
{ 

result = fromDataReader (reader); 
} 
reader.Close (); 
return result; 


} 
// 添 加 模块 数据 
public static int add(ApplicationModule module) 


} 


string sql = "insert into Application Module (ID,NAME,DESCRIPTION, 
URL) values(:id, :name, :description, :url)"; 

var oracle = MyDbUtility.oracle; 

DbCommand command = oracle.GetSqlStringCommand(sql); 
oracle.AddInParameter (command, ":id", DbType.Sstring, module.id); 
oracle.AddInParameter (command, ":name", DbType.Sstring, module. 
name); 

oracle.AddInParameter (command, ":description", DbType.string, 
module. description); 

oracle.AddInParameter (command, ":url", DbType.Sstring, module. 
Ui 

return oracle.ExecuteNonQuery (command); 


/ /修改 模块 数据 
public static int update (ApplicationModule module) 


1 


string sql = "update Application Module set NAME=:name, DESCRIPTION= 
:description, URL=:url where ID=:id"; 

Var oracle = MyDbUtility.oracle; 

DbCommand command = oracle-GetSqlStringCommand (sql); 
oracle.AddInParameter (command, ":id", DbType.Sstring, module.id); 
oracle.AddInParameter (command, ":name", DbType.String, module. 
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name); 

oracle.AddInParameter (command, ":description", DbType.string, 
module. description); 
oracle.AddInParameter (command, 
Url)s> 

return oracle.ExecuteNonQuery (command); 


:url", DbType.String, module. 


} 
// 根 据 模块 ID 删除 模块 
public static int delete(string id) 


string sql = "delete from Application Module where ID=:id"; 
var oracle = MyDbUtility.oracle; 

DbCommand command = oracle.GetSqlStringCommand(sql); 
oracle.AddInParameter (command, ":id", DbType.Sstring, id); 


return oracle.ExecuteNonQuery (command); 


i 
// 从 数据 读 取 器 的 当前 行 数据 生成 一 个 功能 模块 对 象 


private static ApplicationModule fromDataReader (IDataReader reader) 


和 


ApplicationModule module = new ApplicationModule(); 
module.description = reader["description"] .ToString(); 


module.id = reader["id"] .ToString(); 


module.name = reader["name"] .ToString(); 


module.url = reader["url"] .ToString(); 
return module; 


Th 


(3) 为 ApplicationModuleDB 类 编写 单元 测试 代码 并 运行 测试 ， 此 处 省 略 测试 代码 。 


(4) 编写 业务 逻辑 层 代码 ， 此 处 省 略 具 体 代 码 。 


(5) 添加 一 个 页 面 以 显示 数据 和 接受 用 户 输入 ， 此 处 省 略 页 面 的 具体 实现 代码 。 


13.4.6 ”角色 权限 管理 


社保 卡 结算 系统 使 用 的 权限 管理 模块 是 一 种 基于 角色 的 权限 管理 。 一 个 角色 拥有 一 组 
权限 ， 用 户 属于 哪个 角色 就 拥有 哪个 角色 的 权限 。 系 统管 理 员 用 户 可 以 为 其 他 角色 分 配 权 
限 。 角 色 权 限 分 配 功能 可 以 指定 一 个 角色 允许 或 者 禁止 访问 某 个 模块 ， 此 功能 具体 实现 步 


又 如 下 。 


(1) 在 业务 实体 项 目 Account Entity 中 添加 一 个 类 ApplicationModule， 以 描述 角色 权 


限 信息 。 


// 角 色 权 限 业 务实 体 类 ， 描 述 了 一 个 角色 对 一 个 功能 模块 的 权限 


public class RoleRight 
public string role { get; set; } 


public string right { get; set; } //1 表示 拥有 权限 ，0 表示 没有 权限 


public ApplicationModule module { get; set; } 


i 


(2) 在 数据 访问 层 项 目 AccountCheckDAL 中 添加 一 个 类 RoleRightDB， 实现 与 角色 权 


限 相 关 的 数据 访问 功能 。 
public static class RoleRightDB 
{ 
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#region public methods 

/// <summary> 

/// 得 到 某 个 角色 的 权限 列表 

/// </summary> 

/// <param name="role"> 角 色 ID</param> 

/// <returns> 权 限 列 表 </returns> 

public static List<RoleRight> getRoleRight (string role) 
string sql = "select role id, module id, right from role right where 
role id=:role"™; 
var oracle = MyDbUtility.oracle; 
DbCommand command = oracle.GetSqlStringCommand(sql); 
oracle.AddInParameter (command, ":role", DbType.String, role); 
IDataReader reader = oracle.ExecuteReader (command); 
List<RoleRight> list = new List<RoleRight>(); 
while (reader .Read()) 
| 

1ist.RAdd(fromDataReader (reader) ) 7 

} 
reader.Close(); 
return list; 

} 

/// <summary> 

/// 判断 某 个 角色 是 否 可 以 访问 某 个 页 面 

/// </summary> 

/// <param name="role"> 角 色 ID</param> 

/// <param name="page"> 要 访问 的 页 面 </param> 

/// <returns> 是 否 有 权限 访问 </returns> 

public static bool canAccessPage (string role, string page) 


{ 


var module = ApplicationModuleDB.getByUr] (page); 

// 根 据 页 面 URL 得 到 对 应 的 模块 
if (module == null) return false; // 未 打 到 指定 模块 
string sql = "select right from md.role right where role id=:role 
and module id=:module"; 
var oracle = MyDbUtility.oracle; 
DbCommand command = oracle.GetSqlStringCommand(sql1) 
oracle.AddInParameter (command, ":role", DbType.String, role); 
oracle.AddInParameter (command, ":module", DbType.string, module. 


id); 

object o= oracle.ExecuteScalar (command); ”// 得 到 权限 信息 
if (o == null || o == DBNul1.Value) return true; 
return o.ToString() == "1"; 


} 

/// <summary> 

/// 修改 角色 权限 

/// </summary> 

/// <param name="roleRight"> 要 修改 的 角色 权限 数据 </param> 

/// <returns> 修 改 成 功 返 回 1， 否 则 0</returns> 

public static int update (RoleRight roleRight) 

上 
string sql = "update md.role right set right=:right where module id= 
:module and role id=:role"; 
Var oracle = MyDbUtility.oracle; 
DbCommand command = oracle-GetSqlStringCommand (sql); 
oracle.AddInParameter (command, ":role", DbType.String, roleRight. 
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} 


role); 

oracle.AddInParameter (command, ":right", DbType.String, roleRi-— 
ght.right); 

oracle.AddInParameter (command, ":module", DbType.string, 
roleRight. module.id); 

return oracle.ExecuteNonQuery (command); 


/// <summary> 

/// 刷新 角色 权限 ， 删 除 过 期 数据 ， 为 新 角色 和 新 模块 生成 默认 数据 
/// </summary> 

/// <returns> 新 产生 的 角色 权限 数据 记录 数 </returns> 
public static int refreshRoleRight () 


. 


string sql; 

// 删 除 角色 权限 表 中 不 存在 的 角色 数据 

sql = "delete from md.role right where role id not in(select id from 
md.user role)"7 

MyDbUtility.oracle.ExecuteNonQuery (System.Data.CommandTYpe .Text， 
5q1); 

// 删 除 角色 权限 表 中 不 存在 的 模块 数据 

sql = "delete from md.role right where module id not in(select id 
from md.application module) "7 

MyDbUtility.oracle.ExecuteNonQuery (System.Data.CommandType. Text, 


5q1); 

sql = "update md.role right set right="'0' where right<>'0' and 
E> 

MyDbUtility.oracle.ExecuteNonQuery (System.Data.CommandType.Text, 
5q1); 


// 为 新 的 角色 和 新 的 模块 生成 角色 权限 数据 ， 并 赋 默 认 值 为 0 没有 权限 ) 


sql=@"insert into role right (role id,module id,right) 


select r.id as rid,m.id as mid,'0' as right2 
from user role r cross join application module m 
where not exists (select * from role right rr where rr.role id=r.id and 


rr.module id=m.id )"; 


} 


return MyDbUtility.oracle.ExecuteNonQuery (System.Data.Command— 
Type. Text, sql); 


#endregion 

#region private methods 

/// <summary> 

/// 从 数据 读 取 器 当前 记录 产生 一 个 角色 权限 对 象 

/// </summary> 

/// <param name="reader"> 数 据 读 取 器 </param> 

/// <returns> 所 产生 的 角色 权限 对 象 </returns> 

private static RoleRight fromDataReader (IDataReader reader) 


由 
RoleRight result = new RoleRight (); 
result.role = reader["role id"] .ToString() 7 
result.right = reader["right"].ToString(); 
string temp = reader["module id"].ToString(); 
result.module = ApplicationModuleDB .getByID (temp); 
return result; 

#endregion 
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(3) 
(4) 
C5 


13.4.7 


根据 
构 用 户 只 
疗 机 构 列 
的 不 同 用 
角色 进行 


为 RoleRightDB 类 编写 单元 测试 代码 并 运行 测试 ， 此 处 省 略 测试 代码 。 
编写 与 角色 权限 相关 的 业务 逻辑 层 代 码 ， 此 处 省 略 具体 代码 。 
添加 一 个 页 面 以 显示 数据 和 接收 用 户 输入 ， 此 处 省 略 页 面 的 具体 实现 代码 。 


医疗 机 构 权 限 管理 


户 需求 ， 社 保 卡 结算 系统 的 用 户 只 可 以 访问 被 允许 的 医疗 机 构 的 数据 。 医 疗 机 
可 以 访问 本 机 构 的 数据 ， 而 社保 机 构 用 户 则 需要 系统 管理 员 为 其 分 配 可 访问 的 医 
表 。 这 种 权限 管理 方式 称 为 医疗 机 构 权 限 管理 。 根 据 用 户 需求 ， 即 使 同一 个 角色 
户 所 允许 访问 的 医疗 机 构 列表 也 不 相同 ， 因 此 医疗 机 构 权 限 管理 是 基于 用 户 而 非 
控制 的 。 下 面 将 介绍 医疗 机 构 权 限 管理 的 具体 实现 步骤 。 


(1) 在 业务 实体 项 目 Account Entity 中 添加 一 个 类 HospitalRight, 以 描述 用 户 医院 权限 。 
public class HospitalRight 


0 


L 


(2) 
机 构 权 限 


public string userId { get; set; } // 用 户 ID 
public string hospitalNo { get; set; } // 医 疗 机 构 编 码 
public string hostpitalName { get; set; } // 医 疗 机 构 名 称 
public string InstituteNo { get; set; } /1 社保 机 构 编 号 
public string right { get; set; } // 权 限 


// 根 据 医疗 机 构 实体 对 象 设计 各 个 字段 的 值 
public void setHospital (Hospital hospital) 


this.hostpitalName = hospital.Name; 
this.InstituteNo = hospital.InstituteNo; 
this.hospitalNo = hospital .HospitalNo; 


在 数据 访问 层 项 目 AccountCheckDB 中 添加 一 个 类 HospitalRightDB， 实 现 与 医疗 
相关 的 数据 访问 功能 。 


public static class HospitalRightDB 


{ 


/// <summary> 
/// 刷新 用 户 医院 权限 数据 ， 为 新 医疗 机 构 和 新 用 户 插入 数据 
/// </summary> 
/// <returns> 新 添加 的 数据 行 数 </returns> 
public static int refreshHospitalRight() 
{ 
string sql; 
// 将 所 有 权限 既 不 为 0 也 不 为 1 的 非法 数据 全 部 设置 为 0 
sql = "update md.user hospital right set right='0' where right<>'1"' 
and right<>"'0"'"; 
MyDbUtility.oracle.ExecuteNonQuery (System.Data.CommandType. Text, 
sq1); 
// 添 加 新 医 医疗 机 构 和 新 用 户 的 权限 〈 默 认为 0) 
sql = @"insert into md.user hospital right 
(user id,yybm, sbjgbh, right) 
select r.id as user id2,m.yybm,m.sbjgbh,'0' as right2 
from md.web user r cross join md.institution natl mm 
where not exists 
(select * from md.user hospital right wu 
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| 

WE 
VA 
A 
WA 
/// 
A 
/// 
MAA 
A 
/// 
Wi 
pA 


where wu.user id=r.id and wu.yybm=m.yybm and wu.sbjgbh= 


mspijgbn 
return MyDbUtility.oracle.ExecuteNonQuery (System.Data.Command— 
Type. Text, sql); 


<summary> 


根据 用 户 名 和 搜索 关键 字 查找 医院 权限 

</summary> 

<param name="user"> 用 户 ID</param> 

<param name="key"> 搜 索 关键 字 (医院 名 称 或 者 医院 编码 ) </param> 

<param name="byName"> 根 据 名 称 还 是 编码 ，true 名 称 ，false 编码 </param> 
<param name="page"> 分 页 参数 </param> 
<returns> 查 找到 的 医院 权限 列表 </returns> 


<remarks> 
如 果 搜 索 关键 字 (参数 key) 不 为 空 ， 则 查找 指定 用 户 对 符合 条 件 的 医疗 机 构 的 权限 ， 
如 果 搜 索 关键 字 为 空 ， 则 查找 指定 用 户 对 所 有 医疗 机 构 的 权限 


</remarks> 


public static List<HospitalRight> getUserHospitalRight (string user, 
string key, bool byName, PageDataArgument page) 


string sql = null; 

// 如 果 没 有 搜索 条 件 则 直接 查询 

if (string.IsNullOrEmpty (key) ) 

{ 

sql = @"select * from ( 

select tl1.yybm,t1.sbjgbh,t1.right,t2.yymc,rownum as rn 
from md.user hospital right tl inner join md.INSTITUTION 
NATL 七 2 
on tl1.yybm=t2.yybm and tl.sbjgbh=t2.sbjgbh 
where tl.user id=:u0 order by tl1.sbjgbh,tl1.Yyybm) "7 

} 


else 
{ 
// 根 据 医院 名 称 查找 
if (byName) 
| 
sql = @"select * from ( 
select t1 .yybm,t1.sbjgbh,t1.right,t2.yymc,rownumas-— 
rn frommd.user hospital right tl1 inner join md.INSTI- 
TUTION_NATL 七 2 
on tl1.yybm=t2.yybm and tl1.sbjgbh=t2.sbjgbh 
where tl.user id=:u0 and t2.yymc like :key order by 
t1.sbjgbh,t1 .yybm)"; 
} 
// 根 据 医院 编码 查找 
else 


sql = @"select * from ( 
select t1 .yybm,t1.sbjgbh,t1.right,t2.yymc,rownumas— 
rn from md.user hospital right tl inner join md. 
INSTITUTION NATL t2 
on tl1.yybm=t2.yybm and t1.sbjgbh=t2.sbjgbh 
where tl.user id=:u0 and tl.yybm like :key order by 
t1.sbjgbh, t1.yybm) "; 
F 

F 

Var oracle = MyDbUtility.oracle; 

DbCommand command = oracle-GetSqlStringCommand (sql); 

oracle.AddInParameter (command, ":u0", DbType.Sstring, user); 
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// 有 搜索 条 件 则 添加 到 where 子 句 中 
if (!string.IsNullOrEmpty (key) ) 

oracle.AddInParameter (command, ":key", DbType.String, "%"+ key 

和 
if (page.refreshCount) 

page -count = MyDbUtility.getCount (command); 
command.CommandText = sql + " where rn between " 

+ page.min + "and " + page.max; // 生 成 分 页 条 件 
IDataReader reader = oracle.ExecuteReader (Command) 
List<HospitalRight> result = new List<HospitalRight>() 
// 循 环 读 取 数 据 并 添加 到 列表 中 
while (reader.Read()) 

{ 

HospitalRight r = new HospitalRight (); 

r.hospitalNo = reader[0].ToString(); 

r.InstituteNo = reader[1].ToString(); 

r.hostpitalName = reader["yymc"] .ToString () 7 

r.right = reader["right"] .ToString(); 

r.userId = user; 

result.Add (r); 

} 


return result; 


// 得 到 指定 用 户 的 医疗 机 构 权 限 列表 
public static List<HospitalRight> getUserHospitalRight (string user, 
PageDataArgument page) 


return getUserHospitalRight (user, null,false,page); 


public static List<Hospital> getGrantedHospitals (string user,string 
yybm, PageDataArgument page) 


return getHospitalsByRight (user, yybm, true, page); 


// 得 到 指定 用 户 被 授权 的 医疗 机 构 列 表 
public static List<Hospital> getGrantedHospitals (string user,Page-— 
DataArgument page) 


return getHospitalsByRight (user,null,true,page); 

// 得 到 指定 用 户 未 被 授权 的 医疗 机 构 列表 

public static List<Hospital> getUngrantedHospitals (string user,Page- 
DataArgument page) 


return getHospitalsByRight (user,null, false,page); 


/// <summary> 

/// 根据 权限 分 类 得 到 指定 用 户 的 医疗 机 构 列表 

/// </summary> 

/// <param name="user"> 指 定 用 户 </param> 

/// <param name="yybm"> 医 疗 机 构 编码 </param> 
A can"> 是 否 可 以 访问 </param> 
/// <param name="page"> 分 页 参数 </param> 

/// <returns> 得 到 的 医疗 机 构 列表 </returns> 
private static List<Hospital> getHospitalsByRight (string user, string 
yybm, bool can,PageDataArgument page) 

{ 


SErLng x CA Or> 
string sql =null; 
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// 是 否 有 过 滤 条 件 〈 根 据 医院 编码 过 滤 ) 
bool filter=!string.IsNullOrEmpty (yybm);/ 
if (filter) 
sql = " select * from (select yybm,sbjgbh,rownum as rn from 
md.user hospital right"+" where right=:r0 and user id=:u0 


and yybm like :yybm) "; 


else 
sql = "select * from (select yybm,sbjgbh,rownum as rn from 
md.user hospital right "+ " where right=:r0 and user id=: 
u0)™; 
var oracle = MyDbUtility.oracle; 
// 创 建 命令 并 添加 参数 


DbCommand command = oracle.GetSqlSstringCommand(sql); 
oracle.RddInParameter (command, ":r0", DbType.Sstring, r); 
oracle.AddInParameter (command, ":u0", DbType.String, user); 
if (filter) 
oracle.AddInParameter (command, ":yybm", DbType.String, "%" 
+yybm+"%"); 
// 得 到 记录 总 数 
if (page.refreshCount) 
page.count = MyDbUtility.getCount (command); 
// 设 置 分 页 查询 命令 
command.CommandText = sql + " where rn between " + page.min + " and 
"+ page.max; 
// 读 取 数 据 并 构建 实体 对 象 列表 
IDataReader reader = oracle.ExecuteReader (command); 
List<Hospital> result = new List<Hospital>(); 
while (reader.Read()) 
{ 
Hospital item = HospitalDB.getByID (reader[0].ToString(), 
reader[1] .TosString ()); 
result.Add (item); 
} 


return result; 


// 修 改 医院 权限 数据 
public static int update (HospitalRight right) 


{ 


// 创 建 update 语句 

string sql = "update md.user hospital right set right=:r0 where 
user id=:u0 and sbjgbh=:sbjgbh and yybm=:yybm"; 

var oracle = MyDbUtility.oracle; 

/ /创建 命令 ， 添 加 参数 ， 执 行 命令 

DbCommand command = oracle.GetSqlstringCommand (sql); 
oracle.AddInParameter (command, ":u0", DbType.string, right.user- 
Id); 

oracle.AddInParameter (command, ":sbjgbh", DbType.Sstring, right. 
InstituteNo); 

oracle.AddInParameter (command, ":yybm", DbType.string,right. 
hospitalNo); 

oracle.AddInParameter (command, ":r0", DbType.sString, right. 
right); 

return oracle.ExecuteNonQuery (command); 


(3) 编写 单元 测试 以 测试 AccountCheckDB.HospitalRightDB 类 。 此 处 省 略 测 试 代码 。 
(4) 在 业务 罗 辑 层 项 目 AccountCheckBLL 中 添加 一 个 类 HospitalRightBLL， 这 个 类 的 
E 要 代码 是 调用 数据 访问 层 AccountCheckDB.HospitalRightDB 类 的 相应 方法 。 此 处 省 略 
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HospitalRightBLL 类 的 代码 。 
(5) 在 表现 层 项 目 AccountCheckWeb 中 ， 添 加 一 个 页 面 HospitalRightPage.aspx， 此 页 
面 可 以 查看 和 修改 用 户 对 医院 的 权限 ， 页 面 设计 外 观 如 图 13.14 所 示 。 


UserRightMaster master 


了 录 
[ ek 本 介 
费用 二 次 结算 。 ”医疗 机 构 字典 。 ”用 户 权限 管理 
系统 用 户 管理 电 
用 户 角 色 管 理 
功能 模块 管理 
角色 权限 管理 
医院 权限 管理 


| J ; 
设计 ] 口 拆 分 | 回 荐 | [| esp:Content#ont "| 


图 13.14 医院 权限 页 面 设计 外 观 


图 13.14 所 示 HospitalRightPage.aspx 页 面 布局 较为 复杂 ， 下 面 对 页 面 功 能 和 使 用 做 一 
个 简单 说 明 。 从 用 户 列表 中 选择 一 个 用 户 然后 单 击 “ 查 看 ”按钮 , 则 在 页 面 下 方 的 GridView 
中 显示 此 用 户 对 所 有 医疗 机 构 的 权限 。 由 于 医疗 机 构 众 多 ， 从 GridView 列表 中 直接 查找 某 
个 医疗 机 构 很 不 方便 ， 此 时 可 以 通过 输入 医院 编码 或 者 医院 名 称 的 方式 进行 过 滤 ， 快 速 找 
到 某 个 医院 ,可 以 在 GridView 中 一 次 选中 多 个 医疗 机 构 , 对 其 进行 批量 授权 或 者 取消 授权 。 
HospitalRightPage.aspx 页 面 代码 如 下 : 


<%@ Register assembly="AspNetPager" namespace="Wuqi .Webdiyer" tagprefix 
="webdiyer" 当 > 
<asp:Content ID="Contentl" ContentPlaceHolderID="head" runat="server"> 
<$-- 导 入 JavaScript 文件 --%$> 
<script src="../js/jquery-1.3.2.js" type="text/javascript"></script> 
<script src="../js/jquery-ui-1.7.2.js" type="text/javascript"></script> 
<script src="../js/MyUtility.js" type="text/javascript"></script> 
<script type="text/javascript"> 
$(function () { 
// 为 GridView 添加 光 棒 效果 
$('table.gridview') .find("tr") .hover( 
function() { $(this).addClass('hoverRow'); }, 
function() { $(this) .removeClass('hoverRow'); } 
); //$('table') .tr.hover 
$SjlUtility.addButtonClass (); 
// 为 全 选 CheckBox 添加 事件 处 理 程序 
$('#selectall') .change (checkAllChanged); 
Ys //$ (function) 
// 当 全 选 checkBox 状态 发 生 改 变 时 ， 更 改 所 有 数据 行 的 CheckBox 状态 
function checkAllChanged() 
| 


var b = this.checked; 
$ ("input:checkbox", $('table.gridview')) 
-each (function () 
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this.checked = bay 
); 
1 
</script> 
</asp:Content> 
<asp:Content ID="Content2" ContentPlaceHolderID="ContentPlaceHolder1l" 
runat="server"> 
< 用户 州 表 > 
用 户 : <asp:DropDownList ID="userList" runat="server"></asp:DropDown- 
List> 
<asp:Button ID="ok" runat="server" Text=" 查 看 "onclick="ok Click" /> gnbsp; 
每 页 记录 数 <asp:TextBox ID="pageSize" runat="server" Width="40px">10</asp: 
TextBox> 
<asp:Button ID="Buttonl" runat="server"” Text=" 修 改 " onclick="Buttonl 
Re /2 Cir 
筛选 条 件 : 
医院 编码 <asp:TextBox ID="hospitalCode" runat="server"></asp:TextBox> 
医院 名 称 <asp:TextBox ID="hospitalName" runat="server"></asp:TextBox> 
<asp:Button ID="filter" runat="server" Text=" 筛 选 " onclick="filter Click" 
><br /> 
对 选中 的 医院 批量 授权 
<asp:DropDownList ID="rightList" runat="server"> 
<asp:ListItem Selected="True" Value="1"> 人 允许 </asp:ListItem> 
<asp:ListItem Value="0"> 拒 绝 </asp:ListItem> 
</asp:DropDownList>gnbsp; 
<asp:Button ID="Button2" runat="server" Text=" 确 定 " onclick="Button2_ 
Click YY 
br /> <br /> 
<asp:HiddenField ID="hiddenUser" runat="server" /> 
<asp:GridView ID="GridViewl" runat="server" CssClass="gridview" 
AutoGenerateColumns="False" DataKeyNames="InstituteNo,Hospital— 
No” > 
<Columns> 
<asp:BoundField DataField="InstituteNo" HeaderText=" 社 保 机 构 编号 " 
ItemSstyle-Width="100" /> 
<asp:BoundField DataField="HospitalNo" HeaderText=" 医 院 编码 " Item- 
Style-Width="100" /> 
<asp:BoundField DataField="HospitalName" HeaderText=" 医 院 名 称 " Item- 
Style-Width="300" /> 
<asp:TemplateField HeaderText=" 权 限 " ItemStyle-Width="60"> 
<ItemTemplate> 
<asp:Label ID="theRight" runat="server" Text='<$%#Eval ("right") 
-ToString()=="1"2" 有 ":" 无 ” %$>'></asp:Label> 
</ItemTemplate> 
</asp:TemplateField> 
<asp:BoundField DataField="right" Visible="false" /> 
<asp:TemplateField HeaderText=" 操 作 " Visible="false" > 
<ItemTemplate> 
<asp:LinkButton runat="server" ID="operation" Text=" 人 允许 禁止 " 
CommandName="grant deny" /> 
</ItemTemplate> 
</asp:TemplateField> 
<asp:TemplateField > 
<HeaderTemplate> 
<!-- 在 GridVievw 表格 头 部 显示 一 个 全 选 CheckBox--> 
<input type="checkbox" id="selectall" /> 
</HeaderTemplate> 
<ItemTemplate> 
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<asp:CheckBox ID="selectCheckBox" runat="server" /> 

</ItemTemplate> 

</asp:TemplateField> 
</Columns> 
</asp:GridView> 
< 分 页 控件 3> 
<webdiyer:AspNetPager ID="pagerl" runat="server" 

onpagechanged= "pagerl PageChanged"> </webdiyer:AspNetPager> 
</asp:Content> 


(6) 在 HospitalRightPage.aspx 页 面 的 Page_ Load 事件 中 ， 初 始 化 数据 绑 定 ， 包 括 刷新 
医院 权限 数据 和 绑 定 用 户 列表 。 


protected void Page Load (object sender, EventArgs e) 


if (!IsPostBack) 
{ 
HospitalRightBLL.refreshHospitalRight ();  // 刷 新 医院 权限 数据 
bindList() 7 
} 
} 
// 绑 定 用 户 列表 


private void bindList() 
List<WebUser> list= WebUserBLL.getAllUsers(); // 得 到 所 有 用 户 
foreach (WebUser item in list) 
{ 
ListItem li = new ListItem() 
li.Text = item.name + "[" + item.id+"]"; 
li.Value = item.id; 
userList.Items.Add (1i); // 将 用 户 添加 到 下 拉 列 表 中 
} 


userList.SelectedIndex = 0; 
} 
(7) 在 HospitalRightPage.aspx 页 面 的 “查看 ”按钮 的 Click 事件 中 ， 查 找 并 显示 指定 
日 户 对 所 有 医院 的 权限 数据 。 


protected void ok Click(object sender, EventArgs e) 
{ 


| 


hospitalName.Text = ""; 
hospitalCode.Text = ""; 
pagerl.CurrentPageIndex = 1 
bindRights (true); 


; 

/// <summary> 

/// 绑 定 医院 权限 列表 到 GriqView 控件 

/// </summary> 

/// <param name="refreshCount"> 是 否 刷新 总 记录 数 </param> 

private void bindRights (bool refreshCount) 

有 
string user = userList.SelectedValue; // 得 到 下 拉 列 表 选 中 的 用 户 
hiddenUser.Value = user; 
PageDataArgument arg = new PageDataArgument (); 
arg.refreshCount = refreshCount; 
arg-pageIndex = pagerl.CurrentPageIndex - 1; 
arg.pageSize = pagerl .PageSize; 
List<HospitalRight> list = null; 
bool filter = hospitalCode.Text.Length > 0 


. 450 。 


第 13 章 社保 卡 结算 系统 


11 hospitalName.Text.Length > 0; // 是 否 有 过 滤 条 件 


1 (FELLESr) 
{ 


if (hospitalCode.Text.Length > 0) // 根 据 医院 编码 过 滤 


list=HospitalRightBLL.getUserHospitalRight (user, hospital-— 


Code. Text.Trim(), false, arg); 


else 
if (hospitalName.Text.Length > 0) // 根 据 医院 名 称 过 滤 
list=HospitalRightBLL.getUserHospitalRight (user, hospit-— 
alName.Text.Trim(), true, arg); 
} 
else 
list=HospitalRightBLL.getUserHospitalRight (user, arg); 


// 没 有 过 滤 条 件 
if(refreshCount) 
pagerl .RecordCount = arg.count; 
GridViewl .DataSource = list; 
GridViewl .DataBind(); 


(8) 在 分 页 控件 的 PageChanged 事件 中 ， 重 新 绑 定 新 的 一 页 数据 。 


protected void pagerl PageChanged (object sender, EventArgs e) 
{ 

bindRights (false); 
上 


(9) 在 “确定 ”按钮 的 Click 事件 中 ， 批 量 修改 选择 医院 的 权限 。 
protected void Button2 Click(object sender, EventArgs e) 


i 
bool changed = false; 


string right = rightList.SelectedValue; // 有 无 权限 


string user = hiddenUser.Value; // 要 修改 权限 的 用 户 
for (int i = 0; i < GridView]l.Rows.Count; i++) 


// 针 对 GridView 中 所 有 行 循环 
i 
CheckBox c = GridView]l .Rows[i].FindControl ("selectCheckBox") as 
CheckBox; 
if (!c.Checked) continue; // 未 选中 该 行 则 继续 下 一 行 


HospitalRight item = new HospitalRight () 7 
item.right = right; 


item.InstituteNo = GridView1.DataKeys[i] ["InstituteNo"] .ToStr— 


ing(); 


item.hospitalNo = GridView]l .DataKeys[i] ["HospitalNo"] .ToString (); 


item.userId = user; 
HospitalRightBLL.update (item); // 修 改 授权 
changed = true; 
| 
if (changed) 
bindRights (true); 
} 


(10) 在 “筛选 ”按钮 的 Click 事件 中 ， 根 据 指 定 的 医院 编码 或 医院 名 称 筛 选 并 显示 医 


院 权限 数据 。 


protected void filter Click(object sender, EventArgs e) 
{ 
bindRights (true); 
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6 在 浏览 览 i) Ra aspx 页 面 ， 显 示 界 面 如 图 13.15 所 示 。 


用 户 : rr 查看 ”每 页 记录 数 10 此 改 
用 户 前 色 管理 箭 选 条件: 医院 编码 77T 医院 名 称 匡 迁 


角色 权限 管理 所 
dt 371301 277 山东 滨 化 实业 公司 有 5S 
371652 770 滨州 市 瑞 丰 医药 连锁 经 营 有 限 责任 公司 十 六 店 有 区 

371652 yl 滨州 市 瑞 丰 医药 连锁 经 营 有 限 责任 公司 十 四 店 有 4 

371652 3772 滨州 市 博 丰 医药 连锁 经 营 有 限 责任 公司 十 店 有 区 

371699 yy73 滨州 市 王 丰 医药 连锁 经 营 有 限 责任 公司 四 店 有 La 

371699 yd 该 州 市 瑞 丰 连 帧 经 营 公司 一 店 有 

371699 775 读 州 市 璃 丰 连 钠 经 营 有 限 责任 公司 中 心 店 有 5 

有 La 


371699 27T76 滨州 市 计划 生育 指导 中 心 


图 13.15 医院 权限 页 面 


13.4.8 ”用 户 登录 


社保 卡 结算 系统 仅 对 A 市 人 力 资源 与 社会 保障 局 及 定点 医疗 机 构 人 员 开 放 , 使 用 该 系 
统 必须 具有 合法 的 用 户 名 和 密码 ， py 用 户 登 录 成 功 后 ， 需 要 把 登录 信息 保存 
在 Session 中 ， 并 在 其 他 页 面 中 根据 此 信息 进行 用 户 权 限 检 测 和 其 他 操作 。 在 表现 层 项 目 
AccountCheckWeb 中 添加 一 个 Login.aspx 页 面 。 页 面 布局 如 图 13.16 所 示 。 


市 劳动 和 和 社会 保障 局 
人 


图 13.16 ”登录 页 面 


. 452 。 


第 13 章 社保 卡 结算 系统 


在 登录 页 面 Login.aspx 中 ， 为 了 获得 较 好 的 用 户 体 验 ， 用 JavaScript 添加 了 一 些 特效 ， 
如 果 鼠 标 移 动 到 按钮 上 会 变色 ， 鼠 标 移动 到 文本 框 时 文本 框 会 自动 获得 焦点 并 选中 其 中 文 
本 。Login.aspx 页 面 代码 如 下 : 


<head runat="server"> 


<title>A 市 劳动 局 医保 数据 对 账 系统 </title> 
<link type="text/css" rel="Stylesheet" href="css/login.css" /> 
<script src="js/jquery-1.3.2.js" type="text/javascript"></script> 
<script type="text/javascript"> 
$ (function() { 
checkIE6(); 
checkResolution(); 
// 当 鼠标 移动 到 用 户 名 和 密码 文本 框 时 ， 自 动 获 得 焦点 ， 并 选中 其 中 文本 
$ ("input#userName") .mouseover (focusSelect); 
$ ("input#password") .mouseover (focusSelect); 
// 为 登录 按钮 添加 样式 
$('input#login2') .addClass ("ui-button ui-state-default ui-— 
corner-all™"); 
$('input#login2') .hover (function () 
{ $(this) .addCclass ("ui-state-hover"); }, 
function() { $(this) .removeClass ("ui-state-hover"); }); 


} 
); 
// 检 查 浏览 器 版 本 不 支持 IE6 及 以 下 ) 
function checkIE6() { 
if ($.browser.msie) { 
if($.browser.version <= 6) 
alert ("你 使 用 的 浏览 器 太 旧 (IE6 及 以 下 ) 了 ， 请 使 用 新 的 浏览 器 (如 IE 
8, FireFox, Chrome 等 "); 
} 
} 
// 检 查 屏 幕 分 辩 率 
function checkResolution() 
{ 
if(screen.width<1000) 


alert (' 请 将 屏幕 分 辨 改 为 1024 以 上 。'); 
} 
// 使 控件 获得 焦点 ， 并 选中 控件 中 的 所 有 文本 


function focusSelect() { 
thissEtocusl)> 
this.select (); 
Dl 
</script> 
</head> 
<body> 
<form id="forml" runat="server"> 
<div id="main"> 
<div> 
<img src="images/1ogo.jpg"” alt="" /> 
</div> 
<div id="login"> 
用 户 ID <asp:TextBox ID="userName" runat="server" AutoCompleteType= 
"None" > </asp:TextBox> 
密 gnbsp; 人 码 <asp:TextBox ID="password" runat="server" TextMode="Pass- 
word"> </asp:TextBox> 


<asp:Button ID="login2" runat="server" Text=" 登录 " Width="80" 
Height="25" onclick="login2 Click"/> 
</div> 
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</div> 
</form> 
</body> 
在 “登录 ”按钮 的 Click 事件 中 ， 验 证 用 户 输入 的 用 户 名 和 密码 ， 如 果 正 确 ， 则 跳 转 
到 默认 页 面 。 由 于 社保 卡 结算 系统 有 两 类 用 户 : 社会 保障 局 工作 人 员 和 医疗 机 构 工作 人 员 ， 
这 两 类 用 户 登 录 使 用 的 用 户 名 和 密码 也 不 相同 ， 所 以 在 验证 用 户 登 录 时 要 注意 这 一 点 ， 代 
码 如 下 : 


protected void login2 Click(object sender, EventArgs e) 
* 


bool success = false; 
WebUser user=WebUserBLL.getByID (userName.Text); 

// 根 据 输入 的 用 户 名 查找 用 户 信息 
if (user == null) // 未 找到 用 户 


// 判 断 是 否 医疗 机 构 
Hospital h=HospitalBLL .getBYID (userName .Text，Ppassword.Text) ; 
if(h!=nul1) // 是 医疗 机 构 
success=true; 
user=new WebUser(){ id=userName.Text, name=h.Name}; 
user.role = UserRole.HospitalRole; 
WebUtility.instituteId=password.Text; 
} 
else // 找 到 了 用 户 
{ 
if (user.password == password.Text)// 密 码 是 否 正确 
{ 
SCCess = EEUeZ 
WebUtility.instituteId = ""; 
} 


if(success) // 登 录 成 功 则 跳 转 到 默认 页 面 

{ 
Common .WebUtility.currentUser = user; 
Response.Redirect ("default/CheckAccount2.aspx"); 


else // 登 录 失 败 则 提示 错误 信息 


ClientScript.RegisterStartupScript (GetType(), "errorlogin", 
"<script>alert (' 用 户 名 或 密码 错误 ! ') ;</script>"); 


13.5 银行 数据 上 传 


在 社保 卡 结算 系统 中 ,社保 卡 的 消费 数据 由 银行 提供 。 银 行 定期 向 A 市 人 力 资源 与 社 
会 保障 局 提供 一 个 该 时 间 段 内 社保 卡 消费 明细 (一 个 dbf 文件 ), 这 个 文件 通过 社保 卡 结算 
系统 导入 到 A 市 人 力 资源 与 社会 保障 局 的 Oracle 数据 库 中 , 并 以 此 数据 为 基础 实现 结算 和 
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报表 功能 。 在 数据 上 传 时 ， 除 上 传 银行 数据 本 身 以 外 ， 还 要 将 上 传 日 志保 存 起 来 ， 以 备 后 


续 查询 


13:5:1 


数据 访问 层 和 业务 逻辑 层 


数据 访问 层 和 业务 逻辑 层 的 功能 是 将 已 经 上 传 到 服务 器 的 指定 dbf 文件 中 的 数据 导入 
到 Oracle 数据 库 ， 并 记录 上 传 日 志 。 这 个 操作 过 程 需要 用 到 Oracle 数据 库 中 的 两 个 表 : 银 
行 卡 消费 明细 表 MD.BANK_TRANSACTION 和 数据 上 传 日 志 表 MD.UPLOAD LOG, 在 数 
层 编写 两 个 类 分 别 操作 这 两 个 表 。 

(1) 在 数据 访问 层 项 目 AccountCheckDB 中 ， 添 加 一 个 类 DbfTransfer， 用 于 将 dbf 文 
件 中 的 数据 导入 MD.BANK_TRANSACTION 表 。 代 码 如 下 : 


// 将 数据 从 dbf 文件 中 导入 到 Oracle 数据 库 的 md-.Bank transacton 表 
public static class DbfTransfer 


据 访 问 


/// <summary> 
/// 将 数据 从 dbf 文件 中 导入 到 Oracle 数据 库 的 md.Bank transacton 表 
/// </summary> 
/// <param name="file"> 数 据 来 源 dbf 文件 (包含 完整 路 径 和 文件 名 ) </param> 
/// <param name="success"> 成 功 导入 的 数据 行 数 〔 输 出 参数 ) </param> 
/// <param name="failure"> 导 入 失败 的 数据 行 数 ( 输 出 参数 ) </param> 
/// <returns> 是 否 全 部 导入 成 功 </returns> 
public static bool transferData(string file,out int success,out int 
failure) 
! 
Success = 0; 
failure = 0; 


// 创 建 一 个 odbc 连接 对 象 用 于 打开 dbf 文件 
System.Data.Odbc.OdbcConnection oConn = new System.Data.Odbc. 
OdbcConnection(); 
string s = "Driver={Microsoft dBase Driver (*.dbf)};SourceType=DBF; 
SourceDB=" + file+ "; Exclusive=No; Collate=Machine; NULL=NO; 
DELETED=NO; BACKGROUNDFETCH=NO; "7 
oConn .ConnectionString = s; 
System.Data.Odbc.OdbcCommand oCmd = oConn.CreateCommand () 7 
oCmd.CommandText = "SELECT * FROM "+file; 
Database oracle = MyDbUtility.oracle; 
// 生 成 插入 数据 的 SQL 语句 
DbCommand command = 
oracle.GetSqlstringCommand ("insert into md.Bank Transaction " 
+ "(ID, WORKDATE, CORNO, NAME, ACCNO, AMT, FEE, REALAMT, TYPE, 
TERM, TIME, CSERNO) VALUES " 
+"(:id, :workdate, :corno, :name, :accno, :amt, :fee, :realamt, :type, 
:term, :time, :cserno) "); 
// 向 命令 添加 各 个 参数 
oracle.AddInParameter (command, 
oracle.AddInParameter (command, 
oracle.AddInParameter (command, ":corno", DbType.Sstring); 
oracle.AddInParameter (Command，" :name"，DbType.String) 
oracle.AddInParameter (Command，":accno"，DbType.String) 
oracle.AddInParameter (command, ":amt", DbType.Currency); 
oracle.AddInParameter (command, ":fee", DbType.Currency); 
oracle.AddInParameter (command, ":realamt", DbType.Currency); 


:id", DbType.string); 
:workdate", DbType.string); 
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oracle.AddInParameter (Command，" :type"，DbType.String) 
oracle.AddInParameter (Command，" :term"，DbType.String) 
oracle.AddInParameter (command, ":time", DbType.DateTime); 
oracle.AddInParameter (Command，" :cserno"，DbType.String) 7 
DateTime now=DateTime .Now7 

int year= now.Year; 

int nm = 0 

// 生 成 上 传 数据 ID 的 前 级 (根据 当前 日 期 时 间 生成 


String idPrefix = string.Format ("{0:0000}", now.Year) 


+ string.Format ("{0:00}", now.Month) 

+ string.Format("{0:00}", now.Day) + string.Format ("{0:00}", 
now.Hour) 

+ string.Format ("{0:00}", now.Minute) + string.Format("{0: 
00}",now.Second); 


int idSsuffix = 0; // 上 传 数据 ID 的 后 级 
// 上 传 的 dbf 数据 表 中 所 有 列 


string[] columns = {"workdate", "corno","name","accno", "amt", 
"fee", "realamt", "type", "term", "cserno" }; 


try 


{ 


oConn.Open (); 
// 从 dbf 文件 中 循环 读 取 每 行 数据 ， 并 插入 到 oracle 中 
using (IDataReader reader = oCmd.ExecuteReader (CommandBehav- 
ior.CloseConnection )) 
. 
while (reader.Read()) 
{ 


string timeStr = ""; 

int month = 0, day = 0, hour = 0, minute = 0, second = 
0; 

4 


{ 
for (int i = 0; i < columns.Length; i++) 
{ 
oracle.SetParameterValue (command, ":"+columns 
[i], reader[columns[i]]); 
} 
// 当 后 绷 达 到 最 大 值 时 ， 后 绥 清 零 ， 并 重新 生成 前 绥 
if (idSuffix >= 999998) 
idSuffix = 0; 
idPrefix = string.Format ("{0:0000}", now.Year) 
+ string.Format ("{0:00}", now.Month) + string. 
Format ("{0:00}", now.Day) 
+ string.Format ("{0:00}", now.Hour) + 
string.Format ("{0:00}", now.Minute) + 
string. Format("{0:00}", now.Second); 
oracle.SetParameterValue (command, ":id", idPrefix 
+string.Format ("{0:000000}", (idSuffix++) .ToStr— 
ing())); 
// 银 行 提供 的 dbf 文件 中 时 间 为 字符 型 ， 格 式 为 mmddhhmiss 
// 需 要 转换 为 日 期 型 
timeStr = reader["time"] .ToString(); 
month = int.Parse(timesStr.Substring(0, 2)); 
day = int-Parse (timeStr.Substring (2, 2)); 
hour = int.Parse (timeStr.Substring(4, 2)); 
minute = int.Parse (timeStr.Substring(6, 2)); 
second = int.Parse (timeStr.Substring(8, 2)); 
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1 


// 如 果 发 生 异 常 ， 传 输 失败 计数 加 1 


catch 


failure++7 
continue; 


oracle.SetParameterValue (command, ":time", new 
DateTime (year, month, day, hour, minute, second)); 
n = oracle.ExecuteNonQuery (command); 
Success++7 

1 

// 如 果 发 生 异 常 ， 传 输 失败 计数 加 1 


catch (OracleException ex) 


{ 
failuret++; 
} 
catch (Exception ex) 
{ 
failuret+; 


} 


} 
} 
finally 
{ 
oConn.Close (); 
command.Dispose(); 
} 


return failure==0; 


(2) 在 测试 项 目 DalTest 中 为 DbfTransfer 类 创建 单元 测试 并 执行 ， 以 检查 和 改正 
DbfTransfer 中 的 错误 。 

(3) 在 数据 访问 层 项 目 AccountCheckDAL 中 添加 一 个 类 UploadLogDB， 以 添加 和 查 
询 数 据 上 传 日 志 。 代 码 如 下 : 

public static class UploadLogDB 


{ 


/// <summary> 
/// 添加 一 个 数据 上 传 日 志 
/// </summary> 
/// <param name="1og"> 要 添加 的 日 志 </Param> 
/// <returns> 添 加 的 日 志 数量 </returns> 
public static int add(UploadLog log) 
// 构 建 insert 语句 
string sql = "insert into MD.UPLOAD LOG " 
+"(User id, total record, correct record, error record)" 
+" values (:userid, :total, :correct, :err)"; 
Var oracle = MyDbUtility.oracle; 
/ /创建 一 个 命令 ， 并 添加 各 个 参数 
DbCommand command = oracle.GetSqlSstringCommand(sql); 
oracle.AddInParameter (command, ":userid",DbType.Sstring,1og. 
userId); 
oracle.AddInParameter (command, ":total", DbType.Int32, log.total— 
Record); 
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(4) 在 测试 项 目 DalTest 中 为 UploadLogDB 类 创建 单元 测试 并 执行 ， 以 检查 和 改正 


} 

/// 
VA 
/// 
/// 
/// 
大 7 


oracle.AddInParameter (command, ":correct", DbType.Int32, log. 
correctRecord); 

oracle.AddInParameter (command, ":err", DbType.Int32, log. Error- 
Record); 

return oracle.ExecuteNonQuery (command); // 执 行 命令 


<summary> 

根据 时 间 范 围 查 询 上 传 日 志 情 况 

</summary> 

<param name="date"> 日 期 范围 </param> 

<param name="page"> 分 页 参数 </param> 
<returns> 查 询 到 的 符合 条 件 的 日 志 列表 </returns> 


public static List<UploadLog> getByDate (DateRangeArg date，PageData- 
Argument page) 


. 


string sql = "select upload id, user id, upload date," 
+" total record,correct record,error record, rownum as rn " 
+" from MD.UPLOAD LOG " 
+" where Upload Date between :datel and :date2"; 
var oracle = MyDbUtility.oracle; 
DbCommand command = oracle.GetSqlStringCommand (sql); 
oracle.AddInParameter (command, ":datel", DbType.DateTime, date. 
from); 
oracle.AddInParameter (command, ":date2", DbType.DateTime, date. 
to); 
// 如 果 需 要 ， 则 刷新 数据 总 行 数 
if (page.refreshCount) 
page.count = MyDbUtility.getCount (Command) 
// 构 建 分 页 查询 语句 
command .CommandText = "select * from ("+ 
sql + ") where rn >"+tpage.pageIndex*page.pageSize 
+ " and rn<=" + (page.pageIndex+1) * page.pageSize; 
IDataReader reader = oracle.ExecuteReader (command); 
/ /循环 读 取 数据 ， 创 建 对 象 列表 
List<UploadLog> list = new List<UploadLog>(); 
while (reader.Read()) 
{ 
UploadLog log = new UploadqLog () 
log.id = Convert.ToInt32 (reader["upload id"]); 
log.userId = Convert.ToString (reader["user id"]); 
log.date = Convert.ToDateTime (reader["upload date"]); 
log.totalRecord = Convert.ToInt32 (reader["total record"]); 
log.correctRecord = Convert.ToInt32 (reader ["correct 
record"]); 
log.errorRecord = Convert.ToInt32 (reader["error record"]); 
list.Add (log); 
| 
reader.Close (); 
return list; 


DbfTransfer 中 的 错误 。 测 试 代码 如 下 : 


[TestClass () ] 
public class UploadLogDBTest 


了 
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[TestMethod()] 
public void addTest () // 测 试 添加 
{ 
UploadLog log = new UploadLog () 
{ userId = "admin", totalRecord = 5000, 
correctRecord = 4900, errorRecord = 100 }; 


日 志 


// 创 建 一 个 上 传 日 志 


int actual = UploadLogDB.add (1og) ; 


Assert.AreEqual (1, actual); // 方 法 应 该 返回 1 
| 
[TestMethod] 
public void getByDateTest () 
f 
PageDataArgument page = PageDataArgument .all; 
page.refreshCount = true; // 得 到 所 有 数据 并 获取 总 数 
// 设 定 日 期 范围 
DateRangeArg date = DateRangeArg.between2Date ("2010-4-1", "2010- 
A= 0) 
var list = UploadLogDB .getByDate (date,page); 
Assert.IsTrue (list.Count > 0); 
Assert.IsTrue (list.Count == page.count); 
Assert.IsNotNull (list[0]); // 列 表 中 的 数据 不 为 空 
} 


13.5.2 ”数据 上 传 页 面 


在 数据 上 传 页 面 中 ， 用 户 可 以 选择 一 个 dbf 文件 ， 将 文件 上 传 到 服务 器 ， 然 后 把 数据 
从 dbf 文 件 中 转 入 到 Oracle 数据 库 中 。 由 于 银行 提供 的 数据 文件 较 大 ， 上 传 和 导入 需要 一 
定时 间 ， 在 实现 数据 上 传 和 导入 功能 时 ， 需 要 充分 考虑 这 两 个 问题 。 实 现 数据 上 传 和 导入 


页 面 的 具体 步骤 如 下 。 


(1) ASPNET Web 应 用 程序 对 上 传 文件 大 小 有 限制 ， 默 认 情况 下 只 允许 上 传 小 于 2M 
的 文件 。 但 银行 提供 的 社保 卡 消费 数据 dbf 文件 大 都 超过 2M， 需 要 修改 web.config 配置 文 


件 中 system.web 结 点 的 两 个 选项 以 允许 上 传 大 文件 。 
<system.web> 


<!--maxRequestLenght 限制 上 传 文件 大 小 ， 以 KB 为 单位 --> 
<!-- 人 允许 执行 请 求 的 最 大 时 间 限 制 ， 单 位 为 秒 --> 


<httpRuntime maxRequestLength="10240" executionTimeout="120"/> 


</system.web> 


(2) 添加 一 个 页 面 UploadData.aspx， 该 页 面 主要 包含 一 个 文件 上 传 控件 、 一 个 按钮 和 


一 个 显示 上 传 进度 的 动画 图 片 。 页 面 布局 如 图 13.17 所 示 。 
在 UploadData.aspx 页 面 中 , 默认 情况 下 动画 图 片 不 显示 ,只 有 当 数 据 J 


E 在 上 传 和 导入 


时 动画 才 显示 。 在 页 面 上 用 一 个 隐藏 字段 HiddenField 来 标识 是 否 有 数据 ] 
此 标志 来 判断 是 否 显示 动画 图 片 。UploadData.aspx 页 面 代码 如 下 : 


E 在 上 传 ， 并 以 


<asp:Content ID="Content2" ContentPlaceHolderID="ContentPlaceHolderl" 


runat="server"> 
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NromMacter 


Ve 
用 户 权限 管理 


i 
邮 生 下定 下 长 鞭 章 大 
MiddenField - wlosdinglss 
千 看 部 握 上 传 日 志 
瞪 示 : 如 果 数 据 量 较 大 ， 数 据 上 传 和 所 入 会 天 要 几 分 神 的 对 间 ， 请 耐心 等 待 。 
| 文件 已 经 上 传 到 服务 器 ， 正 在 导入 到 数据 库 ， 请 等 待 ---.-- 


| 


EL 上 传 并 导入 
[result] 了 
1 一 
避 训 + ] 口 激 分 癌 源 [4[(asp:ContenttContent?) ol 


图 13.17 数据 上 传 页 面 设计 视图 


<h3> 银 行 数据 上 传 和 导入 </h3> 
<asp:HiddenField runat="server" ID="uploadingFlag" Value="0" /><!-- 是 否 有 
数据 正在 上 传 --> 
<br /> 提示 : 如 果 数 据 量 较 大 ， 数 据 上 传 和 导入 会 需要 几 分 钟 的 时 间 ， 请 耐心 等 待 。 
<br /><hr /> 
<div id="loading" runat="server" visible="false"> 
<span style="font-weight:bold; color:#833;"> 
文件 已 经 上 传 到 服务 器 ， 正 在 导入 到 数据 库 ， 请 等 待 . ..... </span><br /> 
<img src="../images/loading.gif" alt=" 数 据 正在 处 理 " style="height: 200px; 
width: 500px" /> 
</div> 
<asp:FileUpload ID="FileUploadl" runat="server" CssClass="upload"/> 
&nbsp; &nbsp; 
<asp:Button ID="Buttonl" runat="server" Text=" 上 传 并 导入 " onclick= 
Buttonl C1lick™” /> 
<br /><!-- 显 示 上 传 结果 或 者 错误 提示 --> 
<asp:Label ID="result" runat="server" Text="" ForeColor="Red" Font-— 
Size="Large"> </asp:Label> 
</asp:Content> 


(3) 根据 用 户 需 求 ， 本 项 目 只 能 上 传 dbf 文件 ， 在 “上 传 并 导入 ”按钮 的 客户 端 Click 
事件 中 ， 用 JavaScript 对 上 传 文件 类 型 进行 检测 。 


// 检 测 上 传 文件 是 否 为 dbf 文件 

function checkDbf() { 
var file = $('input:file') [0]; // 得 到 上 传 控件 
var fname = file.value; // 得 到 上 传 文件 名 


// 用 正则 表达 式 检查 文件 名 是 否 以 dbf 结尾 ， 如 果 不 是 则 提示 错误 
if (/.dbf$/.test(fname)) { 
return true; 


} 

else { 
alert (' 只 能 上 传 dbf 文件 。'); 
return false; 

上 
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(4) 在 “上 传 并 导入 ”按钮 的 服务 器 端 Click 事件 中 ， 编 写 代码 完成 数据 的 上 传 和 导 


入 功能 。 由 于 数据 文件 较 大 , 将 其 导入 Oracle 数据 库 需 要 一 定时 间 , 为 了 使 页 面 尽早 返回 
使 浏览 器 端 能 够 看 到 结果 ， 使 用 一 个 单独 的 线程 对 数据 进行 导入 ， 并 在 浏览 器 端 使 用 


JavaScript 不 断 刷 新 数据 导入 的 进度 。 


protected void Buttonl Click(object sender, EventArgs e) 


// 为 了 防止 用 户 在 此 次 上 传 未 结束 前 再 上 传 数据 ， 将 上 传 控件 隐藏 
FileUploadl.Visible = false; 
Buttonl .Visible = false; 
uploadingFlag.Value = "1"; // 设 置 上 传 标志 为 1 
loading.Visible = true; // 显 示 上 传动 画 
string path=Server.MapPath("../upload"+"\\"+FileUpload] .FileName); 
if (File.Exists (path)) 

File.Delete (path); 
FileUploadl1 .SaveAs (path); // 保 存 文件 
/ /创建 一 个 单独 的 线程 进行 数据 导入 
Thread thread = new Thread (new ParameterizedThreadStart (transfer 
Data)); 
thread.Start (path); 


/// <summary> 

/// 将 dbf 文件 中 的 数据 导入 到 Oracle 数据 库 

/// </summary> 

/// <param name="path"> 要 导入 的 文件 路 径 </param> 
private void transferData (object path) 


{ 


由 


int success=0, failure=0; // 导 入 成 功 和 失败 的 数目 
bool b = DbfTransferBLL.transferData (Path.ToString()， 
out success, out failure); // 执 行 数据 导入 


UploadLog log = new UploadLog(); 

log.userId = WebUtility.currentUser.id; 

log.errorRecord = failure; 

log.correctRecord = success; 

log.totalRecord = failure + success; 

UploadLogBLL.add (10g); // 添 加 数据 上 传 日 志 

// 在 缓存 中 添加 数据 上 传 状态 ， 浏 览 器 端 将 通过 JavaScript 调用 Web Service 查询 此 

状态 

UploadSstate state= new UploadState () 

{ finished = true, succeededRecord = success, 
failedRecord = failure, totalRecord = failure + success }; 

Cache.Insert (Constants.CacheUploadState, state, null, 
DateTime.Now.AddMinutes (5), TimeSpan.Zero); 


上 述 代码 中 用 到 一 个 类 UploadState， 这 个 类 用 于 描述 数据 上 传 状态 ， 类 的 代码 如 下 。 


public class UploadState 


{ 


public bool finished = false; // 上 传 和 导入 是 否 已 经 完成 
public int totalRecord=0; //qbf 文件 中 总 的 记录 数 
public int succeededRecord=0; // 导 入 成 功 记录 数 

public int failedRecord=0; // 导 入 失败 记录 数 


(5) 添加 一 个 Web 服务 ， 返 回 数据 上 传 和 导入 状态 ， 这 个 Web 服务 被 浏览 器 端 通过 


AJAX 方式 调 有 


o 
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// 查 询 上 传 数据 情况 
[WebService (Namespace = "http://tempuri.org/")] 
[WebServiceBinding (ConformsTo = WsiProfiles.BasicProfilel 1)] 
[System.ComponentModel .ToolboxItem(false)] 
[System.Web.Script.Services.ScriptServicel] 
public class UploadQueryService : System.Web.Services.WebService 
{ 
static Random r = new Random(); 
[WebMethod] 
[ScriptMethod] 
public UploadState refreshUploadState () 
if (Context.Cache [Constants.CacheUploadState] == null) 
return null; 
return Context.Cache [Constants.CacheUploadState] as Uploadstate; 


上 


(6) 在 数据 上 传 页 面 UploadData.aspx 中 ， 编 写 JavaScript 代码 以 AJAX 方式 不 断 调 
Web Service 以 查询 数据 导入 状态 ， 当 数据 导入 结束 后 ， 隐 藏 动画 控件 并 显示 导入 结果 。 


<script type="text/javascript"> 


var myTimer; // 定 时 器 ， 定 时 刷新 数据 导入 状态 
var label = '#<%=result.ClientID %>'; // 上 传 结果 控件 ID 
Var progressBar = '#<%=loading.ClientID%$>'; // 上 传动 画 控件 ID 
var uploadingFlag = '#<%=uploadingFlag.ClientID®>'; 

// 上 传 标志 控件 ID 


$(function () { 
$ ('#<%=Buttonl.ClientID%>') .click (checkDbf);  // 检 测 上 传 文件 类 型 
var flag = $ (uploadingFlag) .val (); 
if (flag == "1") 
myTimer=setInterval (refreshstate, 3000); 


i //$ (function()) 
// 刷 新 数据 上 传 状 态 ， 当 数据 上 传 时 此 方法 每 隔 3 秒 被 调用 一 次 
function refreshState() { 
$.ajax( 
{ 
type: “POST”, 
datas i 
url:"UploadQueryService.asmx/refreshUploadState", 
dataType: "json", 
contentType: "application/json; charset=utf-8", 
success:getResult 
} 
); 
b 
// 取 得 数据 上 传 的 结果 
function getResult (result) { 
if (!result) return; 
if (i!result.d) return; 
var r = result.d; 
/ /数据 上 传 已 经 完成 ， 则 显示 上 传情 况 汇总 
if (r.finished) { 


$ (uploadingFlag) .val ("0"); // 清 除数 据 上 传 标志 
$ (progressBar) .hide(); // 隐 藏 上 传动 画 
if (myTimer) 

clearInterval (myTimer); // 清 除 定时 器 
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var 七 = "总 共 上 传 数据 " + r.totalRecord + "<br/>" 
+" 其 中 成 功 导入 :" + r.succeededRecord+"<br/>" 
+ "导入 失败 : " + r.failedRecord; 
$ (label) .html (t); 
} 


</script> 


本 UploadData.aspx 页 面 ， 运 行 界面 如 图 13.18 所 示 。 


市 劳动 局 社保 卡 结算 对 账 系 桃 - Werills Firefor 
编辑 下 ) 查看 WD 历史 人 G) 书签 @ 工具 中 帮助 中 


[ Wy wi 
医保 账目 核对 。 银行 数据 上 传 “数据 查询 统计 虽 用 二 次 结算 医疗 机 构 字典 ”用 户 权限 管理 
银行 数据 上 传 和 导入 


提示 ， 如 果 数据 量 较 大 ， 数 据 上 传 和 导入 会 需要 几 分 钟 的 时 间 ， 请 耐心 等 待 


文件 已 经 上 传 到 服务 器 ， 正 在 导入 到 数据 库 ， 请 等 待 


图 13.18 数据 上 传 运行 界面 


ei 


13.5.3 ”查询 数据 上 传 日 志 


根据 用 户 需 求 ， 用 户 可 以 根据 时 间 范 围 查询 银行 数据 上 传 日 志 。 这 个 功能 较为 简单 ， 


其 中 数据 访 
的 设计 和 实 


问 层 和 业务 逻辑 层 的 功能 已 经 在 13.1 节 中 实现 ， 本 节 将 介绍 表现 层 Web 页 面 


现 。 


(1) 在 表现 层 项 目 AccountCheckWeb 中 添加 一 个 页 面 UploadLogPage.aspx。 


(2) 页 图 


面 上 放置 两 个 TextBox 用 于 输入 日 期 ,一 个 Button 用 于 执行 查询 ,一 个 GridView 


用 于 显示 查询 结果 。 为 了 方便 用 户 输入 日 期 ， 此 页 面 使 用 jQuery UI 的 DatePicker 对 
ASPNET 的 TextBox 控件 进行 扩展 ， 页 面 代码 如 下 : 


<asp:Content ID="Contentl" ContentPlaceHolderID="head" runat="server"> 
<g-- 导 入 所 需要 的 css 文件 和 JavaScript 文件 --%> 
<link href="../css/ui-lightness/jquery-ui.css" rel="stylesheet" 
type="text/css" /> 
<script src="../js/jquery.js" type="text/javascript"></script> 


<script src=" 


./js/jquery-ui.js" type="text/javascript"></script> 


<script src="../js/MyUtility.js" type="text/javascript"></script> 
<script type="text/javascript" language="javascript"> 


$(function () { 


// 为 gridview 添加 光 棒 效果 
$('table.gridview') .find("tr") .hover( 
function () { $(this).addClass('hoverRow'); }, 


。463 。 


第 3 篇 项 目 实战 


function () { $(this) .removeClass('hoverRow'); } 
); //$('table') .tr.hover 
// 为 两 个 日 期 TextBox 添加 扩展 
$('#<%=datel .ClientID$>') .datepicker (); 
$('#<%$=date2.ClientID$>') .datepicker (); 
Hh //$ (function) 
</script> 
</asp:Content> 
<asp:Content ID="Content2" ContentPlaceHolderID="ContentPlaceHolder1l" 
runat="server"> 
<h3> 数 据 上 传 日 志 </h3> 
时 间 范 围 : <asp:TextBox runat="server" id="datel"/> 
至 <asp:TextBox runat="server" ID="date2" /> 
<asp:Button runat="server" ID="button1"” Text=" 查 询 " onclick="buttonl_ 
Cek /><br /> 
<asp:GridView runat="server" ID="gridl" CssClass="gridview" AutoGenerate 
Columns="false"> 


<Columns> 

<asp:BoundField HeaderText=" 上 传 用 户 " DataField="UserID" ItemStyle-Width= 
ia 人 

<asp:BoundField HeaderText=" 上 传 日 期 " DataField="date" ItemStyle-Width= 
“SO 


<asp:BoundField HeaderText=" 总 记录 数 " DataField="TotalRecord" ItemStyle- 
Width="120" /> 

<asp:BoundField HeaderText=" 导 入 成 功 记录 数 " DataField="correctRecord" 
ItemSstyle-Width="120" /> 

<asp:BoundField HeaderText=" 导入 失败 记录 数 " DataField="errorRecord" 
ItemSstyle-Width="120" /> 

</Columns> 

</asp:GridView> 

<webdiyer:AspNetPager ID="pager" runat="server" onpagechanged="pager 
PageChanged"> 

</webdiyer:AspNetPager> 

</asp:Content> 


(3) 在 页 面 后 台 代 码 中 ， 为 “查询 ”按钮 的 Click 事件 和 分 页 控件 的 PageChanged 事 
件 编写 代码 ， 查 询 数据 并 绑 定 到 GridView 控件 中 。 


protected void button1l Click(object sender, EventArgs e) 
! 


bindData (true); 
/// <summary> 
/// 将 数据 上 传 日 志 绑 定 到 GridView 
/// </summary> 
/// <param name="refreshCount"> 是 否 需 要 刷新 总 记录 数 〈 以 更 新 分 页 控件 的 总 页 数 》 
</param> 
private void bindData (bool refreshCount) 
{ 
// 得 到 要 查询 的 日 期 范围 
DateRangeArg date = DateRangeArg .between2Date (datel] .Text, date2.Text); 
/ /创建 分 页 查询 参数 
PageDataArgument page = new PageDataArgument (); 
page.refreshCount = refreshCount; 
page.pageSize = pager.PageSize; 
page.pageIndex = pager.CurrentPageIndex - 1; 
// 执 行 查询 ， 并 绑 定 数据 
var list = UploadLogBLL .getByDate (date, page); 


。464 。 


第 13 章 社保 卡 结算 系统 


if (refreshCount) 

pager.RecordCount = page.count; 
gridl.DataSource = list; 
gridl.DataBind(); 


protected void pager PageChanged(object sender, EventArgs e) 
{ 
bindData (false); 


(4) 运行 UploadLogPage.aspx 页 面 ， 运 行 界面 如 图 13.19 所 示 。 


社保 卡 结 划 对 账 系统 


Ge 
asp. net inaningeontainer 0 加" 且 "-M- 安 - 科 - 因 


IE 


医保 账目 核对 。 银行 数据 上 伟 数据 查询 统计 费用 二 次 结算 医疗 机 移 字 典 用 户 权限 和 理 


数据 上 传 日 志 
时 间 范 围 ，|2010-04-01 至 2010-04-30 查询 
上 传 用 户 上 传 日 期 总 记录 数 导入 成 功 记录 数 。 ”导入 失 赂 记录 数 


admin 2010-4-27 8:24:38 5000 4900 100 


admin 2010-4-27 8:24:41 5000 4900 100 


图 13.19 数据 上 传 日 志 查询 页 面 


13.6 医疗 机 构 对 应 


在 医保 卡 结算 系统 中 ， 需 要 根据 不 同 的 医疗 机 构 进行 统计 和 结算 。 在 银行 的 医保 卡 消 
费 数据 中 ， 使 用 商户 号 和 终端 号 来 标识 不 同 机 构 ， 但 是 在 A 市 人 力 资源 与 社会 保障 局 的 
Oracle 数据 库 中 ， 使 用 社保 机 构 编码 和 医院 编码 来 标识 医疗 机 构 。 社 保 卡 结算 系统 需要 在 
这 两 种 编码 之 间 建 立 对 应 关系 ， 才 能 实现 数据 共享 。 


13.6.1 实体 类 设计 


在 建立 银行 商户 与 医疗 机 构 对 应 关系 时 ， 涉 及 两 个 业务 实体 : 银行 商户 和 医疗 机 构 。 
在 业务 实体 项 目 Account Entity 中 添加 两 个 类 以 描述 这 两 个 实体 。 


/// <summary> 


/// 医疗 机 构 信息 
/// </summary> 
public class Hospital 


public string InstituteNo { get; set; } /1 社保 机 构 编 码 
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public string HospitalNo { get; set; } // 医 院 编码 
public string Name { get; set; } // 医 院 名 称 

下 

/// <summary> 

/// 此 类 用 于 描述 银行 数据 中 的 商户 信息 

/// </summary> 

public class BankShop 

{ 


public string ShopNo { get; set; } // 商 户 号 
public string TerminalNo { get; set; } // 终 端 号 
public string Name { get; set; } // 商 户 名 称 


public Hospital CorrespondingHospital { get; set; } // 对 应 的 医疗 机 构 


13.6.2 ”数据 访问 层 和 业务 逻辑 层 


数据 访问 层 需要 实现 医疗 机 构 信 息 查询 、 商 户 信 息 查 询 ， 以 及 建立 和 修改 二 者 之 间 的 
对 应 关系 。 商 户 数 据 存储 在 MD.BANK_TRANSACTION 表 中 ， 医 疗 机 构 数据 存储 在 
MDJINSTITUTION NAIL 表 中 ， 二 者 对 应 关系 存储 在 MD.Institute Dictionary 表 中 。 为 了 
提高 类 的 内 聚 性 、 适 当 控制 类 的 大 小 ， 用 3 个 类 实现 与 这 3 个 表 相 关 的 数据 访问 。 
(1) 添加 一 个 HospitalDB 类 ， 实 现 医疗 机 构 相 关 的 数据 访问 功能 。 代 码 如 下 : 
public static class HospitalDB 
下 
#region 数据 库 表 名 和 列 名 
internal const string HospitalCodeColumn YM 
internal const string HospitalNameColumn "YYMC"; 
internal const string InstituteCodeColumn = "SBJGBH"; 
internal const string HospitalTable = "MD.INSTITUTION NATL"; 
#endregion 
/// <summary> 
/// 根据 医院 编码 和 社保 机 构 编码 得 到 医院 信息 
/// </summary> 
/// <param name="id"> 医 院 编码 </param> 
/// <param name="sbjgbh"> 社 保 机 构 编码 </param> 
/// <returns> 得 到 的 医院 数据 </returns> 
public static Hospital getByID(string id, string sbjgbh) 
{ 


string sql = string.Format ("select {0},{1},{4} from {2} where 
aang Ta Eo 
HospitalCodeColumn, HospitalNameColumn, HospitalTable, id, 
InstituteCodeColumn, sbjgbh); 
Var list = getList (sql); 
if (list.Count = 0) return null; 
return list[0]; 
} 
/// <summary> 
/// 得 到 所 有 医疗 机 构 数 据 
/// </summary> 
/// <param name="page"> 分 页 参数 </param> 
/// <returns> 得 到 的 指定 页 面 的 医疗 机 构 数 据 </returns> 
public static List<Hospital> getAllHospitals (PageDataArgument page) 
{ 


string sql = string.Format ("select * from(select a.*,rownum 
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| 


a Fn 


+"from (select {0},{1},{3} from {2} order by {0}) a) ", 
HospitalCodeColumn, HospitalNameColumn, HospitalTable, Insti 
tuteCodeColumn); 
if (page.refreshCount) 
page.count = MyDbUtility.getCount (sql); 
sql += "where rn between " + page.min + " and " + page.max; 
return getList(sql); 
} 
// 根 据 select 语句 进行 查询 并 返回 医院 列表 
private static List<Hospital> getList(string select) 
{ 
Database oracle = MyDbUtility.oracle; 
DbCommand command = oracle.GetSqlStringCommand (select); 
List<Hospital> list = new List<Hospital>(); 
using (IDataReader reader = oracle.ExecuteReader (command)) 
{ 
/7 循环 读 取 数据 ， 并 根据 各 个 字段 的 值 创建 Hospital 对 象 
while (reader.Read()) 
f 
Hospital entity = new Hospital (); 
entity.HospitalNo = reader[HospitalCodeColumn] .To- 
String()s 
entity.Name = reader[HospitalNameColumn] .ToString(); 
entity.InstituteNo = reader[InstituteCodeColumn] .ToStr-— 
ing(); 
list.Add(entity); 
L 


return list; 


(2) 添加 一 个 BankShopDB 类 ， 实 现 与 银行 商户 相关 的 数据 访问 功能 。 


public static class BankShopDB 


{ 


#region 数据 库 中 字段 名 称 
private const string ShopNoColumn = "CORNO"; 


Private const string TerminalNoColumn = "TERM"; 
private const string NameColumn = "NAME"; 
#endregion 


/// <summary> 
/// 取得 未 建立 对 应 关系 的 商户 列表 没有 对 应 医院 的 商户 》 
/// </summary> 
/// <param name="arg"> 分 页 参数 </param> 
public static List<BankShop> getUnassignedInstitutes (PageDataArgument 
arg) 
{ 
/ /构建 查 询 语句 ， 是 一 个 展 套 查询 和 连接 查询 
string sql = ”select * from " 
+"(select ta.*,rownum as rn from " 
+"( select distinct Corno, term,Name from md.Bank Transaction 
nt 
+" where not exists 
+ " ( select YYBM from md.institute dictionary b " 
+" where b.corno=a.corno and b.term=a.term )" 
+" order by corno,term )ta)™; 
if (arg.refreshCount) 
arg.count = MyDbUtility.getCount (sql); 
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sql += " where (rn between " + arg-min + " and " + arg.max + ") "7 
return getList(sql); 
上 
/// <summary> 
/// 得 到 所 有 商户 列表 (不 管 建立 和 未 建立 对 应 关系 ) 
/// </summary> 
/// <param name="arg"> 分 页 参数 </param> 
public static List<BankShop> getAllInstitutes (PageDataArgument arg) 
{ 
/ /构建 查询 用 的 select 语句 
string sql = "select * from 
+"(select a.*,rownum as rn from 
+"(select distinct Corno, term,Name from md.Bank Transaction " 
+" order by corno,term) a ) "; 
if (arg.refreshCount) 
arg.count = MyDbUtility.getCount (sql); 
sql += " where In between " + arg.min + " and " + arg.max; 
// 执 行 查 询 ， 得 到 列表 
List<BankShop> list = getList(sql); 
foreach (BankShop shop in list) 
{ 
shop.CorrespondingHospital = InstituteDictionaryDB.getCorres-— 
pondingHospital (shop); 
} 


return list; 


b 


/// <summary> 

/// 得 到 已 经 建立 对 应 关系 的 商户 列表 

/// </summary> 

/// <param name="arg"> 分 页 参数 </param> 

/// <returns> 查 询 得 到 的 列表 </returns> 

public static List<BankShop> getAssignedInstitutes (PageDataArgument 
arg) 


| 
// 构 建 查询 用 的 select 语句 
string sql = "select * from " 
+" (select corno,term,yybm, sbjgbh, rownum as rn from institute 
dictionary) "7 
if (arg.refreshCount) 
arg.count = MyDbUtility.getCount (sql); 
sql += " where (rn between " + arg.min + " and " + arg.max + ") 
Database oracle = MyDbUtility.oracle; 
/ /创建 一 个 数据 库 命 令 
DbCommand command = oracle.GetSqlstringCommand (sql); 
List<BankShop> list = new List<BankShop>(); 
using (IDataReader reader = oracle.ExecuteReader (command)) 


{ 


着 
了 


// 循 环 读 取 数据 ， 根 据 读 到 的 医院 编号 和 银行 商户 号 得 到 对 应 的 实体 对 象 
while (reader.Read()) 
. 
Hospital hospital = HospitalDB.getByID(reader[2] .ToStr-— 
ing() ,reader[3] .ToString()); 
BankShop shop = getBankShopByID (reader[0] .ToString()， 
reaqder[1] .ToString()) > 
if (shop == null) continue; 
shop.CorrespondingHospital = hospital; 
list.Add (shop); 
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1 
return list; 
下 
/// <summary> 
/// 根据 商户 号 和 终端 号 得 到 商户 信息 
/// </summary> 
/// <param name="shop"> 商 户 号 </param> 
/// <param name="term"> 终 端 号 </param> 
public static BankShop getBankShopByID(string shop, string term) 
{ 
string sql = string.Format ("select Corno, term,Name from md.Bank 
Transaction " 
+ " where corno="'{0}"' and term="' {1}"' and rownum=1", shop, term); 
List<BankShop> list = getList(sql); 
1f (List.Count > 0) 
return list[0]; 
else 
return null; 
1 
/// <summary> 
/// 根据 select 语句 查询 数据 并 且 创 建 商户 列表 
/// </summary> 
private static List<BankShop> getList(string select) 
{ 
Database oracle = MyDbUtility.oracle; 
DbCommand command = oracle.GetSqlStringCommand (select); 
List<BankShop> list = new List<BankShop>(); 
using (IDataReader reader = oracle.ExecuteReader (command)) 


while (reader.Read()) 

{ 
BankShop entity = new BankShop(); 
entity.ShopNo = reader [ShopNoColumn] .ToString(); 
entity.TerminalNo = reader[TerminalNoColumn] .ToString(); 
entity.Name = reader[NameColumn] .ToString(); 
list.Add(entity); 


return list; 


(3) 添加 一 个 InstituteDictionaryDB 类 , 此 类 功能 为 建立 和 维护 银行 商户 与 医疗 机 构 之 
间 的 对 应 关系 ， 并 根据 对 应 关系 查找 另 一 方 。 代 码 如 下 : 


public static class InstituteDictionaryDB 


{ 


#region 数据 库 表 和 字段 名 称 
private const string ShopNoColumn = "CORNO"; 


private const string TerminalNoColumn = "TERM"; 

private const string HospitalCodeColumn = "YYBM"; 

private const string InstituteCodeColumn = "SBJGBH"; 

private const string InstituteDictionaryTable="MD.Institute Dictio- 
Dary"7 

#endregion 

// 得 到 与 指定 商户 对 应 的 医院 


public static Hospital getCorrespondingHospital (BankShop shop) 


{ 
Database oracle = MyDbUtility.oracle; 
string sql = string.Format ("select {0}, {4} from {1} where {2}=:corno 
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} 


and {3}=:term",HospitalCodeColumn, InstituteDictionary Table, 
Shop No Column, TerminalNoColumn,InstituteCodeColumn); 
/ /构建 数据 库 命令 并 添加 参数 
DbCommand command = oracle-GetSqlStringCommand (sql); 
oracle.AddInParameter (command, ":corno", DbType.String, shop.Sh-— 
opNo); 
oracle.AddInParameter (command, ":term", DbType.String, shop.Ter-— 
minalNo); 
// 执 行 命令 ， 读 取 数 据 ， 生 成 医疗 机 构 列 表 
using (IDataReader reader = oracle.ExecuteReader (command)) 
{ 
if (reader.Read()) 
t 
string yybm = reader[0].ToString(); 
string jgbh = reader[1].TosString(); 
return HospitalDB.getByID(yybm, jgbh); 
else 
return null; 


} 


// 得 到 与 医院 对 应 的 商户 列表 
public static List<BankShop> getCorrespondingShop (string hospital) 


{ 


} 


string shopNo = ""; 
string terminal = 
List<BankShop> list new List<BankShop> (); 

Database oracle = MyDbUtility.oracle; 

// 构 建 查询 用 的 select 语句 

string sql = string.Format ("select {0},1{1} from {2} where {3}=: 
yybm", ShopNoColumn, TerminalNoColumn , InstituteDictionary Table, 
HospitalCodeColumn); 

/ /创建 一 个 命令 并 添加 参数 

DbCommand command = oracle.GetSqlStringCommand(sql); 
oracle.AddInParameter (command, ":yybm", DbType.Sstring,hospital); 
using (IDataReader reader = oracle.ExecuteReader (command)) 


{ 


// 循 环 读 取 数据 ， 创 建 BankShop 对 象 

while (reader.Read()) 

下 
ShopNo=reader[0] .ToString(); 
terminal = reader[1].ToString(); 
BankShop shop = new BankShop () 

{ ShopNo = shopNo, TerminalNo = terminal }; 
list.Add (shop); 
} 


return list; 


/// <summary> 

/// 保存 医疗 机 构 与 商户 之 间 的 对 应 关系 

/// </summary> 

/// <param name="shop"> 商 户 对 象 ( 包 含 其 对 应 医院 ) </param> 
public static int saveRelation (BankShop shop) 


{ 


Database oracle = MyDbUtility.oracle; 
// 先 删除 原 有 对 应 关系 


string sql = string.Format ("delete from {0} where {1}=:corno and 
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{2}=:term", 

InstituteDictionaryTable,ShopNoColumn, TerminalNoColumn); 
DbCommand command = oracle.GetSqlSstringCommand (sql); 
oracle.AddInParameter (command, ":corno", DbType.Sstring, shop. 
ShopNo); 
oracle.AddInParameter (command, ":term", DbType.String, shop.Term 
inalNo); 
oracle.ExecuteNonQuery (command); 

// 创 建新 的 对 应 关系 
sql=string.Format (" insert into {0} ({1},1{2},1{3},{4}) values 

(:corno, :term, :yybm, :sbjgbh)", InstituteDictionaryTable, 

ShopNoColumn, TerminalNoColumn, HospitalCodeColumn, Institu-— 

teCodeColumn); 
command = oracle.GetSqlStringCommand (sql); 
oracle.AddInParameter (command, ":corno", DbType.String, shop. 
ShopNo); 
oracle.AddInParameter (command, ":term", DbType.String, shop. 
TerminalNo); 
oracle.AddInParameter (command, ":yybm", DbType.String, shop. 
CorrespondingHospital .HospitalNo); 
oracle.AddInParameter (command, ":sbjgbh", DbType.string, shop. 
CorrespondingHospital.InstituteNo); 
return oracle.ExecuteNonQuery (command); 

} 
/// <summary> 
/// 删除 指定 商户 的 对 应 关系 
/// </summary> 
/// <param name="corno"> 商 户 号 </param> 
/// <param name="term"> 终 端 号 </param> 
public static int deleteRelation(string corno, string term) 
| 
Database oracle = MyDbUtility.oracle; 
string sql = string.Format ("delete from {0} where {1}=:corno and 
{2}=:term", InstituteDictionaryTable, ShopNoColumn, TerminalNoCo- 
lumn); 
DbCommand command = oracle.GetSqlStringCommand (sql); 
oracle.RddInParameter (command, ":corno", DbType.String, corno); 
oracle.AddInParameter (command, ":term", DbType.String, term); 
return oracle.ExecuteNonQuery (command); 
} 
/// <summary> 
/// 自动 对 应 功能 ， 在 名 称 相同 的 医疗 机 构 和 银行 商户 之 间 建立 对 应 关系 
/// </summary> 
public static int autoCorrespond () 
string sql = 
@"insert into institute dictionary (corno, term,yybm, sbjgbh) 
select corno,term,yybm from 
( 
select distinct b.corno,b.term,a.yybm,a.sbjgbh 
from institution natl a inner join bank transaction b 
on a.yymc = b.name 
where not exists 
(select * from institute dictionary i 
where i.corno=b.corno and i.term=b.term) a 
Database oracle = MyDbUtility.oracle; 
DbCommand command=oracle.GetSqlStringCommand(sql); 
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return oracle.ExecuteNonQuery (command); 


(4) 业务 由 辑 层 较为 简单 ， 其 功能 为 调用 数据 访问 层 的 相应 方法 ， 此 处 省 略 其 代码 。 


13.6.3 ”医疗 机 构 对 应 页 面 


在 医疗 机 构 对 应 页 面 中 ， 用 户 可 以 建立 、 查 看 和 修改 银行 商户 与 医疗 机 构 之 间 的 对 应 
关系 。 页 面 整体 分 为 两 个 区 域 ， 上 面 的 区 域 用 于 显示 和 删除 对 应 关系 ， 下 面 的 区 域 用 于 创 
建 对 应 关系 ， 在 某 一 时 刻 这 两 个 区 域 仅 有 一 个 可 见 。 页 面 设计 视图 如 图 13.20 所 示 。 
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图 13.20 医疗 机 构 对 应 页 面 设计 视图 


(1) 在 表现 层 项 目 AccountCheckWeb 中 添加 一 个 页 面 InstituteDictionary.aspx， 页 面 布 
局 如 图 13.20 所 示 ， 页 面 代码 如 下 : 


<asp:Content ID="Content2" ContentPlaceHolderID="ContentPlaceHolderl" 
runat="server"> 
<div id="tabpagel"” runat="server"><!-- 这 个 div 为 显示 对 应 关系 的 区 域 --> <br /> 
<asp:RadioButton ID="radiol" runat="server" Text=" 已 对 应 机 构 " Group- 
Name="groupl" Checked="true"/>&gnbsp; gnbsp; 
<asp:RadioButton ID="radio2" runat="server" Text=" 未 对 应 机 构 " Group- 
Name="groupl" />gnbsp;é&nbsp;<asp:Button 
ID="Buttonl" runat="server"” Text=" 查 看 " onclick="Buttonl Click" 
/>gnbsp; 
<asp:Button ID="Button2" runat="server"” Text=" 自 动 对 应 " onclick= 
"Button2 Click" Visible="false" /> 
<br /><br /> 
<!-- 显 示 数 据 的 GridView--> 
<asp:GridView ID="grid" runat="server" CssClass="gridview" 
AutoGenerateColumns="False" DataKeyNames="ShopNo,TerminalNo" 
onrowdeleting="grid RowDeleting" onrowediting="grid RowEditing"> 
<Columns> 
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<xasp:BoundField DataField="ShopNo"” HeaderText=" 商 户 号 "> 
<HeaderStyle Width="120px" /> 
</asp:BoundField> 
<asp:BoundField DataField="TerminalNo" HeaderText=" 终 端 号 "> 
<Headerstyle Width="80px" /> 
</asp:BoundField> 
<asp:BoundField DataField="Name"” HeaderText=" 商 户 名 称 "> 
<HeaderStyle Width="300px" /> 
</asp:BoundField> 
<asp:TemplateField HeaderText=" 对 应 医院 编码 "> 
<ItemTemplate> 
<%#Eval ("CorrespondingHospital .HospitalNo")%> 
</ItemTemplate> 
</asp:TemplateField> 
<asp:TemplateField HeaderText=" 对 应 医院 名 称 "> 
<ItemTemplate> 
<%#Eval ("CorrespondingHospital .Name")%> 
</ItemTemplate> 
</asp:TemplateField> 
<asp:CommandField EditImageUrl="~/images/edit.png" HeaderText=" 
编辑 " 
EditText="" ShowEditButton="true" ButtonType="Image" /> 
<asp:CommandField DeleteImageUrl="~/images/delete.png" Header- 
Text=" 有 删除 " 
DeleteText="" ShowDeleteButton="True" ButtonType="Image" /> 
</Columns> 
</asp:GridView> 
<webdiyer:AspNetPager ID="pagerl" runat="server" 
onpagechanged="pagerl1 PageChanged" ShowCustomInfoSection= 
"Left" 
ShowBoxThreshold="10"> 
</webdiyer:AspNetPager> 
</div> <!--tabpagel 结束 --> 
<!-- 这 个 div 为 编辑 和 创建 对 应 关系 的 区 域 --> 
<div id="editDiv" runat="server"><br /><br /> 
商户 号 : <asp:TextBox ID="shop" runat="server" ></asp:TextBox> 
终端 号 : <asp:TextBox ID="terminal" runat="server"></asp: TextBox> 
&nbsp; tnbsp; 
<asp:Button ID="OK" runat="server"” Text=" 确 定 " onclick="OK Click" 
Visible="false" />&nbsp;&nbsp; 
<asp:Button ID="cancel" runat="server"” Text=" 取 消 " onclick="cancel 
click” /><br /> 
<!-- 显 示 医 院 列 表 --> 
<asp:GridView ID="hospitalGrid" runat="server" AutoGenerateColumns= 
"False™" 
DataKeyNames="HospitalNo" CssClass="gridview" 
onselectedindexchanged="hospitalGrid SelectedIndexChanged"> 
<Columns> 
<asp:BoundField DataField="InstituteNo" HeaderText=" 社 保 机 构 编 号 
" /> 
<asp:BoundField DataField="HospitalNo" HeaderText=" 医 院 编号 " /> 
<asp:BoundField DataField="Name" HeaderText=" 医 院 名 称 " /> 
<asp:CommandField ShowSelectButton="True" SelectText=" 对 应 " 
HeaderText=" 操 作 " /> 
</Columns> 
</asp:GridView> 
<webdiyer:AspNetPager ID="pager2" runat="server" ShowCustomInfo-- 
Section="Left"™" 
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ShowBoxThreshold="10" onpagechanged="pager2 PageChanged"> 
</webdiyer:AspNetPager> 
</div> 
</asp:Content> 


(2) 在 Page_Load 事件 中 ， 加 载 第 一 页 银行 商户 数据 和 医疗 机 构 数据 。 


protected void Page Load (object sender, EventArgs e) 
| 
if (!IsPostBack) 
loadShop (true); 
loadHospital (true); 


} 
/// <summary> 
/// 加 载 一 页 医院 数据 
/// </summary> 
/// <param name="firstTime"> 是 否 第 一 次 加 载 (首次 加 载 时 刷新 总 记录 数 ) </param> 
private void loadHospital (bool firstTime) 
{ 
/ /创建 分 页 检索 参数 并 设置 各 个 属性 
PageDataArgument page = new PageDataArgument (); 
page.pageSize = pager2.PageSize; 
page.pageIndex = pager2.CurrentPageIndex - 1; 
page.refreshCount = firstTime; 
// 执 行 查询 ， 并 绑 定 数据 
List<Hospital> list = HospitalBLL.getAllHospitals (page); 
pager2.RecordCount = page.count; 
hospitalGrid.DataSource = list; 
hospitalGrid.DataBind(); 
} 
/// <summary> 
/// 加 载 一 页 商户 数据 
/// </summary> 
/// <param name="firstTime"> 是 否 第 一 次 加 载 〈 首 次 加 载 时 刷新 总 记录 数 ) </param> 
private void loadShop (bool firstTime) 
{ 
List<BankShop> list=null; 
/ /创建 分 页 检索 参数 并 设置 各 个 属性 
PageDataArgument arg = new PageDataArgument (); 
arg.pageSize = pager?2.PageSize; 
arg.pagelIndex = pager2.CurrentPageIndex - 1; 
arg.refreshCount = firstTime; 
if(firstTime)pagerl.CurrentPageIndex = 1; 
// 加 载 已 对 应 的 商户 还 是 未 对 应 的 商户 
if(radiol.Checked) 
list=BankShopBLL.getAssignedInstitutes (arg); 
else 
list = BankShopBLL.getUnassignedInstitutes (arg); 
if (firstTime) 
{ 
pagerl .RecordCount = arg.count; 


} 
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grid.DataSource 
grid.DataBind(); 


TLE 
(3) 在 两 个 分 页 控件 的 PageChanged 事件 中 ， 加 载 新 的 一 页 数据 。 


protected void pagerl PageChanged (object sender, EventArgs e) 
l loadShop (false); 

J void pager2 PageChanged (object sender, EventArgs e) 
: loadHospital (false); 


(4) 在 第 一 个 GridView 的 RowDeleting 事件 中 ， 删 除 指定 的 对 应 关系 。 


protected void grid RowDeleting (object sender, GridViewDeleteEventArgs e) 
{ 
string shopno = grid.Rows[e.RowIndex] .Cells[0].Text; 
string term = grid.Rowsl[e.RowIndex] .Cells[1] .Text; 
InstituteDictionaryBLL.deleteRelation(shopno, term); 
loadShop (false); 
} 


(5) 在 第 一 个 GridView 的 RowEditing 事件 中 ， 进 入 编辑 视图 ， 并 根据 当前 行 设置 商 
户 号 和 终端 号 。 


protected void grid RowEditing (object sender, GridViewEditEventArgs e) 
{ 
editDiv.Visible = true; 
tabpagel .Visible = false; 
shop.Text = grid.DataKeys [e.NewEditIndex] [0] .ToString (); 
terminal.Text = grid.DataKeys[e.NewEditIndex] [1] .ToString(); 
e.Cancel = true; // 取 消 编辑 事件 
} 


(6) 在 第 二 个 GridView〔 医 院 列表 ) 的 SelectedIndexChanged 中 ， 在 所 选中 的 医院 和 
当前 商户 之 间 建 立 对 应 关系 。 


protected void hospitalGrid SelectedIndexChanged (object sender, Event-— 
Args e) 
{ 
BankShop bankshop = new BankShop(); 
Hospital h= new Hospital(); 
h.HospitalNo = hospitalGrid.SelectedRow.Cells[1] .Text; // 得 到 医院 编码 
h.InstituteNo= hospitalGrid.SelectedRow.Cells[0] .Text; 
// 得 到 社保 机 构 编码 
bankshop.CorrespondingHospital = h; 
bankshop.ShopNo = shop.Text; 
bankshop.TerminalNo = terminal.Text; 
int n = InstituteDictionaryBLL.saveRelation (bankshop); 
// 建 立 对 应 关系 
loadSshop (false); 
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(7) 运行 InstitueDictionary.aspx 页 面 ， 运 行 界面 如 图 13.21 所 示 。 


1 本 机 林 。 个 未 Ha 机 构 。 下 者 


两 户 号 EE FE 对 庙 医院 篇 色 对 启 医 院 名 称 EEC 
a 55000003 。。 过 市 工区 喜 合 大 药 后 ye 深 州 而 演 四 区 天 全 大 克 后 x 
com | 可 E23 下 州 和 三好 大 吉庆 有 站 公司 Hx 

55000002 3 山东 光州 万 寺 光 医 有 有 荫 公司 中 心机 后 。 加 X 

rd 深 州 太 妇 幼 保健 站 x 

mn 活 州 外交 集团 公司 王 秃 袜 x 

2 诅 放 市 平民 中 心机 后 me 活 州 站 平 风 中心 后 x 
2380620028 该 市 次 丰 医药 连 包 径 党 有 限 表 任 公司 二 店 02 庆 州 六 病 丰 医药 过 铀 经 宫 有 对 责任 公司 二 店 加 。 XX 
2380930001 。 55000001 被 个 医院 第 二 门 论 7 交州 和 中 医 哇 吕 二 门 认 x 
2380990003 。 55000001 。 演 放 市 协和 门诊 群 Xx 
2380990005 。 55000001 。。 滨州 市 开 开 区 村 超市 x 


of 0 


图 13.21 医疗 机 构 对 应 页 面 


13.7 账目 核对 


各 个 医疗 机 构 每 隔 一 段 时 间 需 要 到 A 市 人 力 资源 与 社会 保障 局 进行 账目 核对 和 资金 结 
算 ， 以 银行 提供 的 数据 以 依据 ， 统 计 在 本 机 构 产 生 的 消费 数据 ， 并 获得 相应 的 资金 。 


13.7.1 数据 访问 层 和 业务 逻辑 层 


A 市 人 力 资源 与 社会 保障 局 有 一 套 社保 管理 系统 ， 其 中 包含 参 保 人 员 在 医疗 机 构 的 消 
费 数据 。 另 外 ， 如 前 所 述 ， 银 行 也 向 A 市 人 力 资 源 与 社会 保障 局 提供 一 套 由 银行 系统 产生 
的 社保 卡 消费 数据 。 从 理论 上 来 说 ， 这 两 套数 据 应 该 一 臻 。 本 系统 需要 对 这 两 套数 据 进 行 
核对 ， 如 果 存 在 不 一 致 的 数据 ， 则 显示 出 来 以 方便 用 户 查 找 原因 并 改正 。 
于 银行 数据 MD.Bank_ Transaction 表 中 数据 量 很 大 ， 直 接 在 这 个 表 上 进行 查询 和 统 
计 会 花费 较 多 时 间 。 为 了 提高 性 能 ， 可 以 将 待 统 计 的 原始 数据 (通常 为 某 一 医院 某 时 间 范 
围 内 的 数据 ) 复制 到 一 个 临时 表 中 ， 然 后 再 对 临时 表 中 的 数据 进行 查询 和 统计 。 为 了 避免 
冲突 ， 每 个 用 户 所 使 用 的 临时 表 名 应 该 各 不 相同 。 

在 数据 访问 层 项 目 AccountCheckDAL 中 添加 一 个 类 AccountCheckDB， 代 码 如 下 : 


public class AccountCheckDB 
: 


#region Fields 

private string tableName; // 临 时 表 名 称 

private CheckAccountArg checkArg; // 核 对 账目 参数 
#endregion 

#region Constants 

// 创 建 临时 表 的 SQL 语句 

private const string CreateTableSql11 = "CREATE TABLE "7 

private const string CreateTableSql2=@" (ID CHAR(20) NOT NULL PRIMARY 
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KEY ENABLE, 

CORNO VARCHAR2(30), NAME VARCHAR2 (60), ACCNO VARCHAR2 (40) ， 
AMT NUMBER, FEE NUMBER, REALAMT NUMBER, TYPE VARCHAR2 (20), 
TERM VARCHAR2(20), TIME DATE, CSERNO VARCHAR2 (20), 

YYBM varchar2(18), SBJGBH varchar2(10), SFZHM VARCHAR2 (18) ， 
FSSJ DATE, JYJE NUMBER, XM varchar2(20), ERROR FLAG char(1), 
BRQOX varchar2(50))"; 

private const string TemplateTableName = "TEMP BANK CHECK "; 


// 临 时 表 名 称 前 级 
#endregion 
#region Public Methods 
// 进 行 账目 核对 


public void check (CheckAccountArg arg) 
i 
if (string.IsNullOrEmpty (arg.user)) 
throw new ArgumentNullException ("用 户 ID"); 
if(string.IsNullOrEmpty (arg.hospital)) 
throw new ArgumentNullException ("医院 编码 "); 
if (string.IsNullOrEmpty (arg.institute)) 
throw new ArgumentNullException ("社保 机 构 编 号 "); 
checkArg = arg; 
tableName = getTableName (arg.user) ; 


initTable(); // 创 建 临 时 表 

initReportData(); // 初 始 化 临时 表 数 据 

//processReportData(); 

//getHosptialData(); 

getPersonInfo(); // 得 到 人 员 信 息 
// 删 除 临时 表 


public static void deleteTempTables (string user) 
if (string.IsNullOrEmpty (user)) 


throw new ArgumentNullException ("user"); 
MyDbUtility.dropTableIfExists (getTableName (user)); 


// 根 据 用 户 名 得 到 临时 表 名 称 


public static string getTableName (string user) 


return TemplateTableNametuser; 


#endregion 

#region private methods 

/// <summary> 

/// 删除 再 重建 临时 表 以 清除 表 中 数据 
/// </summary> 

private void initTable () 


MyDbUtility.dropTableIfExists (tableName) 

string sql = CreateTableSql1l + tableName + CreateTableSql2; 
Database oracle = MyDbUtility.oracle; 
oracle.ExecuteNonQuery (CommandType.Text, sql); 


/// <summary> 

/// 将 数据 从 MD .BANK _TRANSACTION 中 复制 到 临时 表 中 
/// </summary> 

private void initReportData() 


// 构 建 插入 语句 


string insert = "insert into "+tableName 
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+ "(id,corno,name,accno,amt, fee, realamt, type,™" 
+" term,time,cserno,ERROR FLAG,yybm,sbjgbh)" 
+" select id,corno,name,accno,amt, fee,realamt," 
+" type,term,time,cserno,'0', :yybm, :sbjgbh " 
+" from md.bank transaction " 
+" where audit no is null and corno=:corno and term=:term " 
+" and time between :begin and :end"; 
Database oracle = MyDbUtility.oracle; 
/ /创建 一 个 命令 对 象 并 添加 各 个 参数 
DbCommand command = oracle.GetSqlStringCommand (insert); 
oracle.AddInParameter (command, ":yybm", DbType.String,checkArg. 
hospital); 
oracle.AddInParameter (command, ":sbjgbh", DbType.String,checkArg. 
institute); 
oracle.AddInParameter (command, ":corno", DbType.Sstring); 
oracle.AddInParameter (Command，" :term"，DbType.String) 
oracle.AddInParameter (Command，" :begin"，DbType.DateTime, check- 
Arg.beginDate); 
oracle.AddInParameter (command, ":end", DbType.DateTime,checkArg. 
endDate); 
List<BankShop> list=InstituteDictionaryDB.getCorrespondingShop 
(checkArg.hospital); 
// 将 与 指定 医疗 机 构 相 对 应 的 商户 消费 记录 导入 临时 表 
foreach (BankShop item in list) 
uf 
oracle.SetParameterValue (command, ":corno", item.ShopNo); 
oracle.SetParameterValue (command, ":term", item.TerminalNo); 
int n=oracle.ExecuteNonQuery (command); 


} 


/// <summary> 

/// 删除 相互 抵消 的 数据 ， 包 括 [PCA-PCC]， [PPT-PPC] ， [PCA-PPT] 
/// </summary> 

/// <remarks> 

/// PCR 正常 消费 ，PPT 退货 ，PCC 消费 冲 正 ，PPC 退货 冲 正 

/// </remarks> 

public void processReportData() 


string[] [] pairs={ 
new string[]{"PCA", "PCC"}, 
new string[]{"PPT","PPC"}, 
new string[]{"PCA", "PPT"}}; 
Database oracle = MyDbUtility.oracle; 
// 先 将 数据 插入 到 一 个 临时 表 中 
string sql="insert into temp id table (idl,id2) select a.id as 
Tdl beid as Ld2 fronm ™ 
+tableName +" a, "+tableName 
+" b where a.cserno =b.cserno and upper(a.type)=:typel and 
upper (b.type)=:type2"; 
DbCommand command = oracle.GetSqlstringCommand (sql); 
oracle.AddInParameter (command, ":typel", DbType.string); 
oracle.AddInParameter (command, ":type2", DbType.string); 
for (int i = 0; i < pairs.Length; i++) 


{ 


// 先 删除 临时 表 ， 再 重新 创建 临时 表 ， 以 清空 数据 
MyDbUtility.dropTableIfExists ("temp id table"); 
sql="create table temp id table (idl varchar2 (20) ,id2 
Varchar2 (20) ) "> 

oracle.ExecuteNonQuery (CommandType.Text, sql) 

string a pairs[i][0]; 
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string b = pairs[i][1]; 

oracle.SetParameterValue (command, ":typel", a); 
oracle.SetParameterValue (command, ":type2", b); 
oracle.ExecuteNonQuery (command); 

// 删 除 临时 表 中 存在 的 互相 抵消 的 数据 

sql = "delete from " + tableName + " where id in (select idl from 
temp id table) "7 

oracle.ExecuteNonQuery (CommandType.Text, sql); 

sql = "delete from " + tableName + " where id in (select id2 from 
temp id table)"; 

oracle.ExecuteNonQuery (CommandType.Text, sql); 


} 
MyDbUtility.dropTableIfExists("temp id table"); 
} 
// 设 置 临时 表 中 医疗 机 构 数据 


public void getHosptialData() 


{ 


} 


Database oracle = MyDbUtility.oracle; 

string sql = "update " + tableName + " set yybm =:yybm,sbjgbh= 
:sbjgbh"; 

DbCommand command = oracle.GetSqlStringCommand(sql); 
oracle.AddInParameter (command, ":yybm", DbType.String, checkArg. 
hospital); 

oracle.AddInParameter (command, ":sbjgbh", DbType.Sstring, check-— 
Arg.institute); 

oracle.ExecuteNonQuery (command); 

getHospitalMoney (); 


// 设 置 参 保 人 员 信 息 


private void getPersonInfo() 


{ 


上 


Database oracle = MyDbUtility.oracle; 

DbCommand command = null; 

string sql = null; 

// 设 置身 份 证 号 

sql = "update " + tableName + 
"aset sfzhm=(select sfzhm from dis.card payout b where rownum=1 
and a.accno=b.kyhzh)"; 

command = oracle.GetSqlStringCommand (sql); 

oracle.ExecuteNonQuery (command); 

// 设 置 姓名 

sql = "update "+ tableName + " a set xm=(select xm from md.emp natl 

b where rownum=1 and a.sfzhm=b.grbh)"; 

command = oracle.GetSqlStringCommand (sql); 

oracle.ExecuteNonQuery (command); 

// 设 置 所 属 区 县 

sql="update "+tableName+" temp set brqx = 
+"(select ssqx from md.card account card,md.sbjgqx qx " 
+"where card.kyhzh=temp.accno and qx.sbjgbh=card.sbjgbh and 
rownum=1)"; 

command = oracle.GetSqlStringCommand (sql); 

oracle.ExecuteNonQuery (command); 


// 获 得 社保 管理 系统 中 的 消费 金额 
private void getHospitalMoney() 


{ 


Database oracle = MyDbUtility.oracle; 
DbCommand command = null; 

string sql = null; 

// 修 改 发 生 时 间 和 交易 金额 
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sql = "update " + tableName 
+"aset (fssj,jyje) =( select fssj,jyje from dis.card payout 
I 
+" where b.yybm=:yybm and b.sbjgbh=:sbjgbh " 
+" and substr (cserno, length (a.cserno)-5,6)=substr (yhjyh, 
length (b.yhjyh)-5,6) " 
+"and a.accno=b.KYHZH)"; 
command = oracle.GetSqlSstringCommand (sql); 
oracle.AddInParameter (command, ":yybm", DbType.String, checkArg. 
hospital); 
oracle.AddInParameter (command, ":sbjgbh", DbType.String, check 
Arg.institute); 
oracle.ExecuteNonQuery (command); 
// 设 置 错 误 标 记 
sql = "update " + tableName + @" set error flag ='1' where 
fssj<timet+(-10/ (24*60)) or fssj>time+(10/ (24*60)) 
or nvl(jyje,0)<amt - 0.01 or nvl(jyje,0)>amt+0.01"; 
oracle.ExecuteNonQuery (CommandType.Text, sql); 
} 


#endregion 

2 

// 对 账 数据 检索 参数 

public class CheckAccountArg 

{ 
public string user; // 查 询 用 户 ID 
public string hospital; // 医 疗 编码 
public string institute; // 社 保 机 构 编码 
public DateTime beginDate; // 查 询 开始 日 期 
public DateTime endDate; // 查 询 结束 日 期 


13.7.2 ”对 账 页 面 


在 对 账 页 面 中 , 用 户 可 以 选择 一 个 医疗 机 构 , 查看 某 一 时 间 范 围 内 的 社保 卡 消 费 数据 ， 
并 将 社保 管理 系统 数据 与 银行 数据 进行 核对 。 对 账 页 面 CheckAccount2.aspx 设计 外 观 如 图 
13.22 所 示 。 


Nrem Master 


狼 
mi 中 ang) 银行 数据 上 读 有 二 次 结 芽 。 “区 疗 机 构 宁 册 “外 户 权限 管理 


医疗 机 构 ，[ 环 对 证 习 要 所 医院 编 玛 午 选 ， 
本 


征 ” 大 币 系 入 天 击 100 号 。 医 系 志 话 -12345 号 其 010-12345675。 
Fen 31 studicdenuil com 


1 | 
EEESBIEEZRC t a 


图 13.22 社保 卡 对 账 页 面 设计 外 观 
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在 图 13.22 所 示 的 社保 卡 对 账 页 面 中 ， 可 以 从 下 拉 列 表 中 选择 一 个 医疗 机 构 ， 输 入 一 
个 日 期 范围 ， 查 看 满足 条 件 的 社保 卡 消费 数据 。 由 于 医疗 机 构 数量 众多 ， 为 了 能 
下 拉 列 表 中 找到 特定 医院 ， 在 下 拉 列 表 旁 边 增加 了 根据 医院 编码 进行 筛选 的 功能 。 


户 需求 ， 上 


代码 如 下 : 


快速 从 
根据 用 


户 只 可 以 查看 拥有 权限 的 医疗 机 构 的 数据 ， 所 以 在 绑 定 医疗 机 构 下 拉 列 列表 时 
要 根据 用 户 权限 进行 过 滤 。 
(1) 在 表现 层 项 目 CheckAccountWeb 项 目 中 添加 一 个 页 面 CheckAccount2.aspx， 页 面 


<asp:Content ID="Contentl" ContentPlaceHolderID="head" runat="server"> 
<script type="text/javascript"> 


var dl = '#<%=datel.ClientID %>'; // 开 始 日 期 控件 ID 
var d2 = '#<%=date2.ClientID %>'; // 终 止 日 期 控件 ID 
$ (init); 


function init() { 
$('table.gridview') .find("tr") .hover( 
function() { $(this) .addClass('hoverRow'); }, 
function() { $(this) .removeClass('hoverRow'); } 
); //$('table') .tr.hover 
$SjlUtility.addButtonClass () 7 
$SjlUtility.jQueryDatePickerChinese(); 
createDatePicker(); 
// 为 3 个 打印 按钮 添加 事件 处 理 程序 
$('#printReport') .click (printReport); 
$('#printDetail') .click (printDetail); 
$('#countyButton') .click (printCounty); 


}; 
// 将 两 个 日 期 文本 框 扩展 成 jQuery 日 历 控 件 
function createDatePicker() { 
$(d1l) .datepicker (); 
$(d2) .datepicker (); 


}; 
// 检 测 用 户 输 入 的 日 期 
function checkInput() { 
if ($(d1).val() == ""™ || $(d2) .val() == "") { 


alert (' 必 须 选 择 日 期 ! '); 
return false; 
} 
return true; 
] 7 
// 打 开 汇总 报表 窗口 
function PrintReport () { 
window.showModalDialog("PrintCheckResult.aspx",null,， 
"dialogWidth=600px;dialogHeight=450px; "); 


}; 

// 打 开明 细 报 表 窗 口 

function printDetail() { 
window.showModalDialog ("PrintDetail.aspx", null, 
"dialogWidth=800px;dialogHeight=600px; "); 


} 

// 打 开 区 县 汇总 报表 窗口 

function printCounty() { 
window.showModalDialog ("PrintCountySummary.aspx", null, 
"dialogWidth=600px;dialogHeight=450px;"); 

} 


</script> 
</asp:Content> 
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<asp:Content ID="Content2" ContentPlaceHolderID="ContentPlaceHolder1" 
runat="server"> 


<br /> 医疗 机 构 : <asp:DropDownList ID="hospitalList" runat="server"> 
</asp:DropDownList> 

根据 医院 编码 筛选 : <asp:TextBox ID="searchWord" runat="server"> 
</asp: TextBox> 

<asp:Button ID="filter" runat="server"” Text=" 筛 选 " onclick="filter 
Cliek /<br /> 

日 期 : <asp:TextBox ID="datel" runat="server"></asp:TextBox> 

至 <asp:TextBox ID="date2" runat="server"></asp:TextBox> 
<asp:Button ID="OK" runat="server” Text=" 核 对 账目 " onClientClick= 
"return checkInput ();" onclick="OK Click" /> 

<input type="button" id="printReport"” value=" 汇 总 打印 ” /> 

<input type="button" id="countyButton" value=" 区 县 汇总 "” /> 

<input type="button" id="printDetail" value=" 明 细 打 印 " /><br /> 


<div> 


<asp:GridView ID="gridl" runat="server" CssClass="gridview" Ruto 
GenerateColumns="true"> 
<EmptyDataTemplate> 
<span style="color:Red; font-weight:bold;"> 没 有 符合 条 件 的 数据 
</span> 
</EmptyDataTemplate> 
</asp:GridView> 
<webdiyer:AspNetPager ID="pagerl" runat="server" ShowBoxThreshold="10" 
onpagechanged="pager1l PageChanged" > 
</webdiyer:AspNetPager> 
<div> 
总 计 <asp:Label ID="transactionCount" runat="server" Text="0"> 
</asp:Label> 人 次 ， 
费用 总 额 : <asp:Label ID="totalMoney" runat="server" Text="0"> 
</asp:Label>, 
银联 手续 费 : <asp:Label ID="totalFee" runat="server" Text="0"> 
</asp:Label>, 
应 拨付 金额 : <asp:Label ID="realMoney" runat="server" Text="0"> 
</asp:Label> 
</div> 


</div> 
</asp:Content> 


(2) 在 AccountCheck2.aspx 的 Page Load 事件 中 ， 在 医疗 机 构 下 拉 列 表 中 加 载 当 前 登 
录用 户 有 权 访 问 的 所 有 医疗 机 构 。 


protected void Page Load (object sender, EventArgs e) 


{ 


y 


if (!IsPostBack) 
| 

bindHospitalList(); 
} 


/// <summary> 

/// 根据 用 户 权限 绑 定 医院 列表 

/// </summary> 

private void bindHospitalList() 


{ 
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hospitalList.Items.Clear (); 

// 如 果 当 前 用 户 是 医疗 机 构 用 户 则 只 可 以 查看 自己 的 数据 
if (Common.WebUtility.isMedicineShopUser ()) 
i 
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} 


filter.Visible = false; 
Hospital h = HospitalBLL.getByID (Common.WebUtility.currentUser.id, 
AccountCheckWeb .Common .WebUtility.instituteId); 
ListItem item = new ListItem(h.Name, h.HospitalNo+t","+ 
h.InstituteNo); 
hospitalList.Items.Add (item); 
} 


else 


{ 
// 如 果 当 前 用 户 不 是 医疗 机 构 用 户 而 是 社会 保障 局 用 户 ， 根 据 分 配 的 医院 权限 填充 医院 
列表 
WebUser user=WebUtility.currentUser; 
List<Hospital> list = HospitalRightBLL.getGrantedHospitals 
(user.id, 
searchWord.Text .Trim(), PageDataArgument.all); 
for (int i=0;i<list.Count;i++) 
{ 
Hospital h = list[i]; 
if (h == null) return; 
ListItem item = new ListItem(h.Name, h.HospitalNo + "," + h. 
InstituteNo); 
hospitalList.Items.Add (item); 


(3) 在 “核对 账目 ”按钮 的 Click 事件 中 ， 核 对 满足 指定 条 件 的 数据 ， 并 将 结果 显示 
在 GridView 中 。 


protected void OK Click(object sender, EventArgs e) 


{ 


// 检 测 用 户 输入 合法 性 
DateTime dl = DateTime.Parse (datel.Text); 
DateTime d2 = DateTime.Parse (date2.Text) .AddDays (1) .AddSeconds (-1); 
if (dll d2) return: 
string[] temp = hospitalList.SelectedValue.Split(','); 
// 生 成 对 账 数据 查询 参数 
CheckAccountArg arg=new CheckAccountArg() 
| 

user = Common.WebUtility.currentUser.id, 

hospital = temp[0], 

institute = temp[1], 

beginDate=dl,endDate=d2 
] 
Session [Common.Constants.SessionCheckRrg] = arg; 

// 将 查询 参数 保存 到 Session 

RAccountCheckBLL .AccountCheckBLL bll = 

new 


AccountCheckBLL.AccountCheckBLL (Common .WebUtility.currentUser.id); 


bll.check (arg); // 生 成 对 账 数 据 

// 在 页 面 底部 显示 汇总 数据 

CheckAccountReport report = bll.getSummary(); 
totalMoney.Text = string.Format ("{0:C}", report.amt); 
totalFee.Text = string.Format ("{0:C}", report.fee); 
realMoney.Text = string.Format ("{0:C}", report.realamt); 
transactionCount .Text = report.count.ToString(); 
loadData (true); 


/// <summary> 
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/// 加 载 要 对 账 的 数据 并 显示 在 GridqView 中 
/// </summary> 
/// <param name="firstTime"> 是 否 首次 加 载 〈 首 次 加 载 需 要 刷新 记录 总 数 ) </param> 
private void loadData(bool firstTime) 
{ 
AccountCheckBLL.AccountCheckBLL bll = 
new AccountCheckBLL.AccountCheckBLL (Common.WebUtility. 
currentUser.id); 
PageDataArgument arg = new PageDataArgument (); 
arg.pageIndex=pagerl .CurrentPagelIndex - 1; 
arg.pageSize=pagerl] .PageSize; 
arg.refreshCount=firstTime; 
DataTable table = bll.getAllData( arg); 
if (firstTime) 
pagerl .RecordCount = arg.count; 
gridl.DataSource = table; 
gridl.DataBind(); 
» 


(4) 在 分 页 控件 的 PageChanged 事件 中 ， 绑 定 新 的 一 页 数据 。 


protected void pagerl PageChanged (object sender, EventArgs e) 
. 
loadData (false); 


(5) 在 “筛选 ”按钮 的 Click 事件 中 ， 根 据 用 户 输 入 的 医院 编码 对 医疗 机 构 列 表 进 行 
筛选 ， 并 重新 绑 定 医院 列表 。 

protected void filter Click(object sender, EventArgs e) 

{ 


bindHospitalList(); 
} 


(6) 运行 AccountCheck2.aspx 页 面 ， 运 行 结果 如 图 13.23 所 示 。 


市 劳动 局 社保 卡 结算 对 账 系统 - Werilla Firefox 
文件 中 编辑 引 ) 查看 如 历史) 书签 @) 工具 GD) 帮助 如 


外- © x |) tp /ohost:1282/default/CheckAccount?. aspx ll Er 
CR ed “-@- 


1 ”市 劳动 局 社保 卡 结算 对 账 系统 


医疗 机 构 : 根据 医院 编码 筛选: 血 选 省 
日 期 ， 2009-8-1 到 2009-8-31 核对 账目 汇总 打印 区 县 汇总 明细 打印 
商户 名 称 身份 证 号 姓名 银行 王 户 消费 时 间 交易 全 额 手续 费 结算 金额 
10 “市 东 政 通 社区 门诊 6222114025487263 2009-8-17 8:35:03 4 0.04 3.96 
9 “市 东 政 通 社区 门诊 6222114025968312 2009-8-17 19:38:53 107 1.07 105.93 
5 ”市 东 政通 社区 门诊 6222114001300845 2009-8-19 9:05:45 7.50 0.08 7.42 
6 市 东 政 通 社区 门诊 6222114025523398 2009-8-19 9:38:57 45 0.45 44.55 
8 “市 东 政 通 社区 门诊 6222114001507282 2009-8-19 10:42:50 41 0.41 40.59 
7 市 东 政 通 社区 门诊 6222114001529765 2009-8-19 16:12:58 20 0.20 19.80 
3 “市 东 政 通 社区 门诊 6222114001126836 2009-8-19 19:01:05 20 0.20 19.80 
4 “市 东 政通 社区 门诊 6222114001508348 2009-8-19 19:46:14 35 0.35 34.65 
1 市 东 政通 社区 门诊 6222114001464450 2009-8-23 10:42:31 35 0.35 34.65 
2 市 东 政 通 社区 门诊 6222114001199320 2009-8-23 12:43:42 35 0.35 34.65 


图 13.23 账目 核对 页 面 


。484 。 


第 13 章 社保 卡 结算 系统 


13.8 结算 申请 表 


医疗 机 构 在 与 人 力 资源 和 社会 保障 局 进行 结算 以 前 ， 需 要 打印 一 份 《城镇 基本 医疗 保 
险 门诊 、 药 店 费用 结算 申请 表 》。 根据 数据 汇总 方式 的 不 同 , 打印 三 种 不 同 格式 的 结算 申请 
表 。 在 图 13.23 所 示 的 账目 核对 页 面 中 ， 单 击 “ 汇 总 打印 “明细 打印 ”或 者 “区 县 汇总 ” 
按钮 ， 即 分 别 进入 三 种 不 同 格式 的 申请 表 页 面 。 本 节 将 介绍 结算 申请 表 的 生成 和 打印 。 


13.8.1 汇总 表 


汇总 格式 的 结算 申请 表 列 出 了 指定 医疗 机 构 在 指定 时 间 范 围 内 社保 卡 消费 总 额 和 人 
数 总 计 ， 表 格式 如 图 13.24 所 示 。 


| | 币 村 确认 


基 生 时 未 医疗 保险 门诊 、 药 店 费用 结算 申请 表 (城镇 职工 ) 


o |o 0.00 0 0 0 0 


根据 对 帐 情况， 本 期 中 请 划 所 资金: 元 
医疗 机 构 意见 : 医保 机 构 意见 : 


和 二 和: 

从: 公 经 办 机 构 负 责 入 : 
(公章) (公章 ) 

年 月 日 总 时 日 


a 
rib FT 


图 13.24 结算 申请 表 (汇总 格式 ) 
全 提示 : 本 节 只 介绍 结算 申请 表 的 生成 和 打印 ， 不 涉及 结算 申请 表 的 审核 和 资金 结算 。 图 
13.24 中 所 示 页 面 中 审核 确认 按钮 的 功能 将 在 本 章 18.9 中 实现 。 


(1) 在 业务 实体 层 项 目 Account Entity 中 添加 一 个 类 AccountCheckReport 以 描述 结算 
申请 表 的 各 个 字段 。 


public class CheckAccountReport 


public string county { get; set; } // 区 县 名 称 
public int count { get; set; } // 总 人 数 
public double amt { get; set; } // 消 费 金额 
public double fee { get; set; } // 手 续费 
public double realamt { get; set; } // 实 际 结算 金额 


上 

(2) 在 数据 访问 层 项 目 AccountCheckDAL 中 添加 一 个 类 AccountCheckTable， 此 类 主 
要 功能 是 根据 对 账 时 产生 的 临时 表 中 的 数据 进行 汇总 ， 得 到 各 种 报表 。 

// 根 据 对 账 产生 的 临时 表 中 数据 汇总 得 到 报表 数据 
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public class AccountCheckTable 


1 
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private string table; // 临 时 表 名 称 
public AccountCheckTable (string tempTable) 
{ 
table = tempTable; 
} 
// 生 成 汇总 报表 数据 
public CheckAccountReport getSummary () 
于 


return getCountySummary (nul1) 


// 按 照 参 保 人 员 区 县 进行 汇总 
public List<CheckAccountReport> getAllCountySummary () 


| 
// 得 到 临时 表 中 所 有 不 同 的 区 县 列表 
string sql = "select distinct brqx from "+table; 
IDataReader reader = MyDbUtility.oracle.ExecuteReader (CommandType. 
Text, sql); 
List<string> counties = new List<string>(); 
while (reader.Read()) 
{ 
if (reader[0] != null) 
counties.Add (reader [0] .ToString ()); 
} 
reader.Close (); 
// 针 对 每 个 区 县 ， 得 到 区 县 汇总 数据 
List<CheckAccountReport> list = new List<CheckRccountReport>() 
foreach (string county in counties) 
list.Add (getCountySummary (county)); 
// 添 加 合计 数据 
CheckAccountReport report = getSummary () 
report .county = "合计 : "; 
list.Add (report); 
return list; 
1 
/// <summary> 
/// 得 到 在 此 医院 就 医 的 某 区 县 病人 汇总 数据 
/// </summary> 
/// <param name="table"> 临 时 数据 表 名 </param> 
/// <param name="county"> 区 县 名 称 ， 为 空 则 所 有 区 县 </param> 
private CheckAccountReport getCountySummary (string county) 
{ 
bool filter = !string.IsNullOrEmpty(county);  // 是 否 指定 了 区 县 
string where = filter ? (" where brqx='" + County + "™' ") : ""; 
CheckAccountReport result = new CheckAccountReport () ; 


// 得 到 就 诊 总 人 次 数 ， 如 商户 号 、 终 端 号 、 卡 号 都 相同 ， 则 认为 是 同一 笔 交易 〈 一 人 次 ) 


string sql = "select count (*) from " 
+" (select distinct corno,term,accno,to char (time,'yyyy-mm-dd 
hh24")” 


-From "4 table Where 和 > 
result.count = Convert.ToInt32 (MyDbUtility.oracle.ExecuteScalar 
(CommandType.Text, sql)); 
// 得 到 汇总 金额 
sql = "select sum(nvl (amt,0)),sum(nvl(fee,0)) ,sum(nvl(realamt,0)) 
from " + tabletwhere; 
Var reader = MyDbUtility.oracle.ExecuteReader (CommandType.Text, 
sql)s; 
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if (reader.Read()) 


object o reader[0]; 

result.amt = (o == DBNull.Value ? 0 : Convert.ToDouble(o)); 
oO = reader[1]; 

result.fee = (o == DBNull.Value ? 0 : Convert.ToDouble(o)); 


o = reader[2]; 

result.realamt = (o == DBNull.Value ? 0 : Convert.ToDouble(o)) 
上 
result.county = county; 
return result; 


’ 
// 得 到 核对 正确 的 数据 汇总 
public DataTable getCorrectSummary () 


让 


} 


string sql = "select name as 商户 名 称 , sum (amt) as 交易 金额 , sum (fee) as 
手续 费 , sum (realamt) as 结算 金额 from " + table + " where error flag='0' 
group by name"; 

IDataReader reader=MyDbUtility.oracle.ExecuteReader (CommandType. 
Text, sql); 

DataTable 七 = new DataTable(); 

t.Load (reader) 7 

return 七 7 


// 得 到 核对 错误 的 数据 汇总 
public DataTable getErrorSummary() 


} 


string sql = "select name as 商户 名 称 , sum (amt) as 交易 金额 , sum (fee) as 
手续 费 , sum (realamt) as 结算 金额 from " + table + " where error flag='1' 
group by name"; 
IDataReader reader = MyDbUtility.oracle.ExecuteReader (CommandType. 
Text, sql); 

DataTable 七 = new DataTable(); 

t.Load (reader) 7 

return 七 7 


public DataTable getCorrectData (PageDataArgument pageArg) 


l 
} 


return getData( "0",pageArg); 


public DataTable getErrorDate (PageDataArgument pageArg) 


上 


} 
WA 


WA 
A 
A 
WA 
WA 


return getData( "1",pageArg); 


<summary> 


得 到 银行 数据 


</summary> 

<param name="errorFlag"> 错 误 标志 ，1 错误 ,0 正确 ，* 所 有 数据 </param> 
<param name="arg"> 分 页 参数 </param> 
<returns> 得 到 的 银行 数据 </returns> 


private DataTable getData (string errorFlag,PageDataArgument arg) 


{ 


int min = arg.min; 

int max = arg.max; 

DataTable result = null; 

// 查 询 用 的 基本 Select 语句 

string sql = "select * from " 
+"( select rownum as 序号 ，a.name as 商户 名 称 ,a.sfzhm as 身份 
证 号 。" 
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+" bhb-.xmas 姓名 ,a.accno as 银行 账户 ， a.time as 消费 时 间 , a .amt as 交 


易 金 额 ，" 
+" a.fee as 手续 费 ,a.realamt as 结算 金额 from " 
+ table + " a left outer join emp natl b on a.sfzhm=b.grbh "; 
// 如 果 错 误 标志 不 等 于 *， 则 根据 错误 标记 进行 过 滤 ， 生 成 对 应 的 Where 条 件 
if (errorFlag != "*") 
sql 14= "where (a.error flag="" + errorFlag + "') "; 
sql += " order by a.time "7 
td es 
// 如 果 需 要 刷新 记录 总 数 ， 则 调用 MyDbUtility 工具 类 的 方法 得 到 总 数 
if (arg.refreshCount) 
arg.count=MyDbUtility.getCount (sql); 
// 添 加 分 页 查询 条 件 
sql += " where 序号 between " + min + " and " + max; 
Database oracle = MyDbUtility.oracle; 
DbCommand command = oracle.GetSqlStringCommand (sql); 
/ /执行 分 页 查询 ， 得 到 查询 结果 并 返回 
using (IDataReader reader = oracle.ExecuteReader (command)) 
{ 
result = new DataTable(); 
result.Load (reader); 
} 
return result; 
| 
public DataTable getAllData (PageDataArgument arg) 
{ 


return getData( "*", arg); 


i 


(3) 在 业务 逻辑 层 项 目 AccountCheckBLL 中 添加 一 个 类 AccountCheckBLL, 其 主要 功 
能 为 调用 数据 访问 层 相应 方法 ， 此 处 省 略 其 代码 。 

(4) 在 表现 层 项 目 AccountCheckWeb 中 添加 一 个 页 面 PrintCheckResultaspx， 页 面 设 
计 布 局 如 图 13.24 所 示 ， 页 面 代码 如 下 : 


<head runat="server"> 
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<style type="text/css" media="print" > 

.noprint{display:none;} /* 只 显示 不 打印 */ 
</style> 
<style type="text/css" media="all"> 
body{font-family: 宋 体 ; font-size:12px;} 
table{ font-family: 宋 体 ; font-size:12px; border:solid lpx; } 
</style> 
<script src="../js/jquery.js" type="text/javascript"></script> 
<script type="text/javascript"> 

$ (function() { 


$('#printButton') .click (printReport); // 打 印 按钮 Click 事件 
$ ('#payMoney') .change (checkMoney); // 支 付 金 额 changed 事件 
$('#audit') .click (checkMoney); // 审 核 按 钮 Click 事件 
$('#closeButton') .click (function() { window.close(); }); 

// 关 闭 按钮 Click 事件 


var vl = $('#payMoney') .val (); 
$ ('#payMoneyLabel') .html (v1); 


//$ (function) 
function printReport() { 
window.print (); // 打 印 当前 页 面 


} 
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// 检 查实 际 拨付 金额 不 能 大 于 剩余 拨付 金额 
function checkMoney() { 
Var vl = $("'#payMoney') .val (); 
Var pay = parseFloat (v1); 
var v2 = $('#realMoney') .html (); 
var amt = parseFloat (v2); 
$('#payMoneyLabel') .html (v1); 
if (pay > amt) { 
alert (' 实 际 拨付 金额 不 能 大 于 应 拨付 金额 ! 请 修改 ! ') > 
return false; 
} 
return true; 
} 
</script> 
<title> 城镇 基本 医疗 保险 费用 结算 </title> 
</head> 
<body> 
<form id="forml" runat="server"> 
<div class="noprint"> 
<input id="printButton" type="button" value=" 打 印 表格 " /> 
<input type="button" id="closeButton"” value=" 关 闭 窗口 " /> 
&nbsp; gnbsp; gnbsp; 实际 拨付 金额 
<asp:TextBox ID="payMoney" runat="server"></asp:TextBox> 
<asp:Button ID="audit" runat="server"” Text=" 审 核 确认 " onclick= 
"verify Click" /><br /> 
<!- -错误 信息 --> 
<asp:Label ID="errorMessage" runat="server" Text="" ForeColor= 
"Red"></asp:Label> 
</div> 
<!-- 结 算 申请 表 表 头 开 始 --> 
<span style="font-family: 宋体 ; font-size: 14px; font-weight: bold;"> 
城镇 基本 医疗 保险 门诊 、 药 店 费用 结算 申请 表 (城镇 职工 ) </span><br /><br /> 
<asp:Label ID="hospital" runat="server" Text="Label"></asp:Label><br 
ea 
<asp:Label ID="dateRange" runat="server" Text="Label"></asp:Label><br 
Ww 
<! 一 -结算 申请 表 表 头 结束 -> 
<!-- 此 table 为 申请 表 正文 --> 
<table rules="all" cellspacing="0" cellpadding="4"> 
<tr><td> 人 次 </td><tq> 费 用 总 额 </td> 
<td> 统 筹 基金 支付 </td><td> 账 户 消费 金额 </td> 
<td> 银 联手 续费 </td><td> 应 拨付 金额 </td> 
<tqd> 实 际 拨付 金额 </td></tr> 
<tr> 
<td><asp:Label ID="transactionCount" runat="server" Text="0"> 
</asp:Label></td> 
<td><asp:Label ID="totalMoney" runat="server" Text="0"> 
</asp:Label></td> 
<td>0.00</td><td> 
<asp:Label ID="totalMoney2" runat="server" Text="0"> 
</asp:Label></td> 
<td><asp:Label ID="totalFee" runat="server" Text="0"> 
</asp:Label></td> 
<td><asp:Label ID="realMoney" runat="server" Text="0"> 
</asp:Label></td> 
<td><asp:Label ID="payMoneyLabel" runat="server" Text="0"> 
</asp:Label></td> 
RE 
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<tr><td colspan="7"> 根 据 对 账 情况 ， 本 期 申请 划拨 资金 : 

<asp:Label runat="server" id="applyMoney" Width="150"></asp:Label> 
元 。</td> 

</tr> 

<tr> 

<td colspan="4"> 


<pre> 医疗 机 构 意见 : 


负责 人 : 

(公章 ) 

年 月 日 
</pre></td> 
<td colspan="3"> 


<pre> 医保 机 构 意见 : 


经 手 大 : 
科室 负责 人 : 
经 办 机 构 负责 人 : 


年 月 日 


<tr><td colspan="6"> 
备注 : 本 表 一 式 两 份 ， 医 保 经 办 机 构 、 定 点 门诊 或 药店 各 存 一 份 。<br /> 
附 :城镇 基本 医疗 保险 门诊 、 药 店 费用 结算 花 名 册 。 BF /> 
</td></tr> 
</table> 
操作 员 : <asp:Label ID="userName" runat="server" Text="Label"> 
</asp:Label> 
&nbsp; &nbsp; gnbsp; 打印 时 间 : <%=DateTime.Today.ToShortDateString ()%> 
</form> 
</body> 


(5) 在 PrintCheckResult.aspx 页 面 的 Page_ Load 事件 中 ， 得 到 报表 数据 并 显示 在 相应 
控件 上 。 


protected void Page Load (object sender, EventArgs e) 
1 
if (!IsPostBack) 
{ 
// 设 计 审 核 按钮 的 可 见 性 ， 对 医疗 机 构 不 可 见 
audit.Visible = !Common.WebUtility.isMedicineShopUser (); 
generateReport (); // 生 成 报表 数据 
// 生 成 报表 数据 
private void generateReport () 
{ 
// 如 果 输 入 日 期 不 合法 则 返回 
if (!checkDateArg()) 
return; 
CheckAccountArg arg = Session[Common.Constants.SessionCheckArg] as 


CheckAccountArg; // 得 到 对 账 参数 
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有 


Hospital h = HospitalBLL.getByID(arg.hospital, arg.institute); 


hospital.Text = "医疗 机 构 编码 : " + arg-hospital + "&nbsp; &nbsp; 医 疗 机 构 


名 称 : " + h.Name; 

AccountCheckBLL bl11 = new AccountCheckBLL (Common .WebUtility. 
currentUser.id); 

CheckAccountReport report = bll.getSummary(); // 得 到 汇总 数据 
/ /根据 汇 总 数据 设置 各 个 控件 的 值 

totalMoney.Text = totalMoney2.Text=string.Format ("{0:0.00}", 
report .amt); 

totalFee.Text = string.Format ("{0:0.00}", report.fee); 
realMoney.Text = string.Format ("{0:0.00}", report.realamt); 
payMoney.Text = string.Format("{0:0.00}", report.realamt); 
transactionCount.Text = report.count.ToString(); 
userName.Text = Common.WebUtility.currentUser.name; 


// 检 查 并 显示 日 期 范围 
private bool checkDateArg () 


{ 


; 


// 从 Session 中 获得 对 账 检索 参数 

CheckAccountArg arg = Session[Common.Constants.SessionCheckArg] 

CheckAccountArg; 

if (arg == null) 
errorMessage.Text = "未 能 正确 设置 对 账 参数 ， 不 能 显示 和 打印 报表 。"; 
return false; 

dateRange.Text = "起 始 日 期 : "+arg.beginDate.ToShortDateString () 
+"n&nbsp;i&nbsp; 终 止 日 期 : "+arg-endDate.ToShortDateString () 7 

return true; 


(6) 运行 PrintCheckResult.aspx 页 面 ， 运 行 结果 如 图 13.25 所 示 。 


[faetp:/ocdhost: 1282/ defuult/PrintCheckkesult. asps 


打印 表格 | 关闭 窗口 实际 拔 付 全 额 3496. 50 审核 确认 
上 、 药 店 费 用 结算 申请 表 (城镇 职工 ) 


医疗 机 构 六 码 : yy3 医疗 机 构 名 称 : 二 棉 门 诊 
起 始 日 期 : 2009-8-1 终止 日 期 : 2009-6-31 


FE3EEOEEOEREOCEOIEEOEEEEO 
|i a om sum | [wm [ww 
根据 对 由 情况， 本 其 中 请 妈 近 全 元 
医疗 机 构 意见 ; SR 
人 
WA 得力 机 构 负 责 人 
于 “为! 并 《公章 ) 
年 月 日 
人 
队 : 城 本 亲人 给 门 、 药 用 蛇 苑 各 雷 。 


TE EE 
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图 13.25 ”资金 结算 申请 单 〈 汇 总 格式 ) 运行 页 面 


13.8.2 ”区 县 汇总 表 


as 


结算 申请 表 的 第 二 种 格式 为 区 县 汇总 表 ， 可 以 统计 某 医 疗 机 构 就 诊 的 人 员 来 自 于 哪个 
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城 证 基本 医疗 保险 实用 结算 -Wezills Firefoz 
[ee /odest 292/ dun VPrintComtySemaury. apx 


打印 表格 | 关闭 窗口 
EL 药店 费用 结算 申请 去 (城镇 职工 ) 


医疗 机 构 辆 码 : yy3 ”医疗 机 构 名 称 ; 二 析 门 诊 
起 始 日 期 : 2009-6-1 终止 日 期 : 2009-6-31 


区 县 名 称 | 人 次 | 费用 总 额 | 统筹 葵 全 支付 | 账户 消 虽 全 额 | 根 联 手续 八 | 应 按 付 全 额 
沾 化 县 “|3 92.6 0.00 .6 0.45 92.14 
RE | |se |om 之 2 ED 
Ra |s [x om 加 Te | 这 对 
博 兴 县 |4 212.6 |0.00 212.6 1.08 211.52 
部 平 县 |3 1172 0.00 1172 5.88 1166 .12 
wz | less [om 加 3 |ssm 
mm | [0 | om 0 EE | 人 
滨州 市 |1 0.00 92 0.46 91.54 
合计 : |32 |3514.2 |0.00 3514.2 47.7 3496.5 
根据 对 帐 傅 况 ， 本 期 中 请 烛 扣 站 全 元 
医疗 机 构 意 见 MR 
有 
rN DA 和: 

年 月 日 A 
备注 ; 本 表 一 式 两 从 医保 是 办 机 构 、 定 点 门 闪 或 药 各 存 一 从 
附 : 城 慎 基 本 医疗 保险 门 洽 药店 费 用 结算 花 名 从 


后 作 责 ; 管理 页。 条 乓 柯 辣 ;2010-5-3 


E23 
图 13.26 结算 申请 表 (区 县 汇总 格式 ) 


(1) 在 表现 层 项 目 AccountCheckWeb 中 添加 一 个 页 面 PrintCountySummary.aspx。 此 页 
面 代码 与 汇总 格式 的 结算 申请 表 大 致 相同， 只 是 表 头 下 面 用 一 个 Repeater 控件 循环 显示 了 
所 有 区 县 的 汇总 数据 ， 如 下 代码 所 示 。 


<table rules="all" cellspacing="0" cellpadding="4"> 
<tr><td> 区 县 名 称 </td><td> 人 次 </td><td> 费 用 总 额 </td><td> 统 筹 基金 支付 </td> 
<tqd> 账 户 消费 金额 </td><td> 银 联手 续费 </td><td> 应 拨付 金额 </td> 
</tr> 

<asp:Repeater ID="Repeaterl" runat="server"> 
<ItemTemplate> 

<tr><td><%#Eval ("county") %></td> 

<td><%#Eval ("count") %></td> 

<td><%#Eval ("amt") %></td> 

<td>0.00</td> 

<td><%#Eval ("amt")®%></td> 

<td><%#Eval ("fee")®%></td> 

<td><%#Eval ("realamt")%></td> 

</tr> 

</ItemTemplate> 

</asp:Repeater> 


(2) 在 数据 访问 层 项 目 AccountCheckDAL 中 添加 一 个 类 SbjgDB， 以 更 新 和 获取 社保 
机 构 所 属 区 县 。 


public static class SbjgDB 
{ 


// 更 新 社保 机 构 所 属 区 县 
public static void updateCounty() 


{ 


. 492 。 


第 13 章 社保 卡 结算 系统 


// 表 说 明 : si .si natl 为 社保 机 构 总 表 , md .sbjgqx 为 社保 机 构 所 属 区 县 表 
// 将 si_natl 中 存在 而 sbjgqx 表 中 不 存在 的 社保 机 构 添加 到 sbjgqx 中 
string sql = " insert into md.sbjgqx (sbjgbh, sbjgmc)" 
+" select sbjgbh, sbjgmc from si.si natl " 
+ " where sbjgbh not in (select sbjgbh from md.sbjgqx)"; 
Database db = MyDbUtility.oracle; 
int n=db.ExecuteNonQuery (CommandType.Text, sql); 
if (n == 0) return; 
// 查 询 sbjgqx 中 所 属 区 县 为 空 的 社保 机 构 
sql = "select sbjgbh, sbjgmc, ssqx from md.sbjgqx where ssqx is null"; 
DataTable table = db.ExecuteDataSet (CommandType.Text, sql). 
Tables[0]; 
string sbjgbh, sbjgmc, qx; 
// 生 成 update 语句 
sql = "update md.sbjgqx set ssqx=:qx where sbjgbh=:bh"; 
DbCommand command = db.GetSqlStringCommand (sq1) ; 
db.AddInParameter (command, ":qx", DbType.Sstring); 
db.AddInParameter (command, ":bh", DbType.string); 
foreach (DataRow row in table.Rows) // 循 环 更 新 新 机 构 所 属 区 县 
{ 
sbjgbh = row[0] .ToString() 7 
sbjgmc = Convert.ToString (row[1]); 
qx = getCounty (sbjgmc); 
command.Parameters[":qx"] .Value = qx; 
command.Parameters[":bh"] .Value = sbjgbh; 
db.ExecuteNonQuery (command); 
} 
} 
// 此 数组 包含 了 A 市 所 有 区 县 
static string[] couties={" 滨 城区 ", " 博 兴 县 "," 惠 民 县 ", "无 标 县 "," 阳 信和 县 "， 
" 沾 化 县 ", "邹平 县 ", "经 济 开发 区 ", "高 新 区 "} 
// 根 据 社保 机 构 名 称 判断 社保 机 构 所 属 区 县 
public static string getCounty (string sbjgmc) 
// 数 组 中 已 有 区 县 直接 返回 
foreach (string c in couties) 
if (sbjgmc.IndexOf (c) >=0) 
return c; 
ne oxs 
// 在 社保 机 构 名 称 中 查找 "区 "或 者 "县 "出 现 的 位 置 
if ((qx=sbjgmc.Indexof(' 县 ')) > 0 11 (qx=sbjgmc.Indexof (' 区 ')) > 0) 
{ 
//" 市 "之 后 "县 "之 前 的 文本 即 为 区 县 名 称 ， 返 回 这 个 字符 串 
int city=sbjgmc.Indexof(' 市 '); 
3 City < ON 
return sbjgmc.Substring(0, qx + 1); 
return sbjgmc.Substring(city+1l, qx - city ); 
} 
// 如 果 社 保 机 构 不 包含 县 或 区 字样 ， 则 查找 "A 市 "字样 
if (sbjgmc.IndexOof ("A 市 ") >= 0) 
return "A 市 "; 
throw new ApplicationException ("不 能 判断 这 个 社保 机 构 所 属 区 县 : 
"+sbjgmc); 
} 
// 得 到 所 有 社保 机 构 编号 
public static List<string> getAll() 


string sql = "select distinct sbjgbh from md.institution natl"; 
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并 四 


IDataReader reader= MyDbUtility.oracle.ExecuteReader 
(CommandType.Text, sql); 
List<string> list = new List<string>(); 
while (reader.Read()) 
{ 
list.Add(reader[0] .ToString()); 
3 
return list; 


} 
(3) 在 PrintCountySummary.aspx 页 面 的 Page_ Load 事件 中 ， 生 成 并 显示 报表 数据 。 


protected void Page _ Load (object sender, EventArgs e) 
if (!IsPostBack) 
{ 
SbjgBLL.updateCounty (); // 更 新 社保 机 构 所 属 区 县 
generateReport (); // 生 成 报表 
} 
private void generateReport () 


if (!checkDateArg()) 


return; 
CheckAccountArg arg = Session[Common.Constants.SessionCheckArg] as 
CheckAccountArg; // 得 到 对 账 参数 
// 取 得 医院 信息 并 显示 在 页 面 上 


Hospital h = HospitalBLL.getByID(arg.hospital, arg.institute); 
hospital .Text = "医疗 机 构 编 码 : " + arg.hospital + "&nbsp; gnbsp; 医 疗 机 构 
名 称 : " + h.Name; 

// 生 成 报表 汇总 数据 

AccountCheckBLL bl1 = new AccountCheckBLL (Common .WebUtility. 
currentUser.id); 

List<CheckAccountReport> report = bll.getAllCountySummary (); 

// 将 数据 绑 定 到 Repeater 控件 


Repeaterl]l .DataSource = report; 
Repeaterl .DataBind (); 
userName.Text = Common.WebUtility.currentUser.name; 


13.9 审核 和 结算 


医疗 机 构 提 出 结算 申请 以 后 ， 人 力 资源 和 社会 保障 局 工作 人 员 对 这 些 数据 进行 审核 ， 
行 结算 支付 。 资 金 可 以 一 次 结 清 ， 也 可 以 分 成 若干 次 结 清 。 


攻 


13.9.1 实体 类 设计 


审核 
对 此 


根据 用 户 需求 ， 在 审核 医疗 机 构 的 结算 申请 时 ， 需 要 保存 相关 审核 信息 ， 如 审核 人 、 
期 、 各 种 金额 等 。 医 疗 机 构 的 结算 申请 通过 审核 以 后 ， 人 力 资源 和 社会 保障 局 就 会 
次 审核 费用 进行 结算 。 资 金 可 以 一 次 结 清 ， 也 可 以 分 成 若干 次 结 清 。 由 此 可 以 看 出 ， 
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审核 和 结算 之 间 是 一 对 多 的 关系 ， 一 次 审核 可 以 包含 一 到 多 次 结算 ， 而 一 次 结算 必然 只 此 
于 一 次 审核 ,与 此 相关 的 实体 类 有 两 个 :审核 信息 类 AuditInfo 和 审核 支付 类 AuditPayment。 
两 个 类 的 代码 如 下 : 


public class AuditInfo 


{ 
public 
public 
public 
public 
public 
public 
public 
public 
public 
public 
public 


string 


auditNo{get; set;} 


DateTime auditDate{get;set;} 


string 
string 
string 
string 
double 
double 
double 
double 
string 


auditUser{get; set;} 
yybm{get; set;} 
sbjgbh{get; set;} 
yymc{get;set;} 

amt {get; set;} 

fee{get; set;} 

realamt {get; set;} 
payMoney { get; set; } 
cancel { get; set; } 


// 剩 余 金 额 〈 此 交 审 核 尚未 结算 的 金额 ? 


public double moneyRemained 


{ 
} 


get { return realamt - payMoney; } 


public class AuditPayment 


. 

public 
public 
public 
public 
public 
public 
public 
public 
i 


string 


auditNo { get; set; } 


int payNo { get; set; } 


DateTime payDate { get; set; } 


double 
string 
string 


payMoney { get; set; } 
payUser { get; set; } 
cancel { get; set; } 


const int NewPayNo = -1; 
AuditPayment () 


payNo = NewPayNo; 


} 


13.9.2 ”数据 访问 层 和 业务 逻辑 层 


// 审 核 编号 

// 审 核 日 期 

// 审 核 人 
// 医 疗 编码 

// 社 保 机 构 编号 
// 医 院 名 称 
// 消 费 金额 

// 手 续费 

// 实 际 结算 金额 
// 已 经 支付 金额 
// 撤 销 标志 (审核 是 否 被 撤销 ) 


// 审 核 编 号 

// 结 算 支 付 顺 序号 

// 支 付 日 期 
// 支 付 金额 
// 支 付 用 户 
// 撤 销 标 志 

// 新 增 的 资金 支付 序号 


通过 与 用 户 交流 ， 得 知 审核 和 结算 是 密切 相关 的 两 个 操作 。 通 常 医疗 机 构 提 出 某 个 时 


间 段 的 结算 申请 后 ， 人 力 资源 和 社会 保障 局 工作 人 员 审核 这 个 时 间 段 的 费 上 


金 结算 。 数 据 访 问 层 和 业务 逻辑 层 的 主要 功能 是 添加 及 查询 审核 和 支付 数据 ， 根 据 用 户 需 


， 并 且 进 行 资 


求 ， 审 核 和 支付 数据 不 能 被 修改 和 删除 ， 但 是 审核 可 以 被 撤销 。 
如 前 所 述 ， 审 核 和 结算 支付 之 间 是 一 对 多 的 关系 ， 一 次 审核 可 以 包含 一 到 多 次 支付 ， 


而 一 次 支付 必然 只 


属于 一 次 审核 ,为 了 标识 同一 次 审核 所 包含 的 多 次 支付 之 间 的 顺序 关系 ， 


第 一 次 支付 都 被 分 配 一 个 顺序 号 ， 第 一 次 支付 的 顺序 号 为 1， 第 二 次 为 2， 依次 类 推 。 
(1) 在 数据 访问 层 项 目 AccountCheckDAL 中 添加 一 个 类 AuditPayDB， 这 个 类 实现 了 
与 支付 相关 的 数据 访问 功能 。 代 码 如 下 : 
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// 结 算 支付 数据 访问 类 
public static class AuditPayDB 
/ /数据 库 表 名 和 表 中 各 个 字段 名 称 
static string table = " Audit Payment "7 
static string columns =" Audit No,Pay No,Pay Date,Pay Money,Pay User, 
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Cancel "2 
/// <summary> 
/// 根据 审核 编号 得 到 审核 信息 
/// </summary> 
/// <param name="audit"> 审 核 编号 </param> 
public static List<AuditPayment> getByAudit (string audit) 
{ 
string sql = " select " + columns +" from "+ table 
+ " where Audit No=:audit0 order by pay no"; 
var oracle = MyDbUtility.oracle; 
Var command = oracle.GetSqlStringCommand (sql); 
oracle.AddInParameter (command, ":audit0", DbType.Sstring, audit); 
Var reader = oracle.ExecuteReader (command); 
var list = fromReader (reader); 
reader.Close (); 
return list; 
加 
/// <summary> 
/// 添加 支付 信息 
/// </summary> 
/// <param name="pay"> 要 添加 的 支付 信息 </param> 
/// <remarks> 
/// 此 方法 只 应 从 AuditDB.adq() 方 法 中 调用 ( 当 增 加 新 的 支付 时 〉， 
/// 不 应 该 从 页 面 中 或 其 他 代码 中 调用 ， 以 免 产 生 数 据 不 一 致 。 
/// </remarks> 
internal static int add(AuditPayment pay) 
{ 
string sql = "insert into " + table +" (" + columns 
+ ") values (:audit0 no,:pay no,sysdate, :money0, :user0,'0')"; 
if (pay.payNo == AuditPayment .NewPayNo) 
pay.payNo = newPayNo (pay.auditNo); 
var oracle = MyDbUtility.oracle; 
Var command = oracle.GetSqlstringCommand (sql); 
oracle.AddInParameter (command, ":audit0 no", DbType.string, pay. 
auditNo); 
oracle.AddInParameter (command, ":pay_no", DbType.Int32, pay. 
payNo); 
oracle.AddInParameter (command, ":money0", DbType.Double, pay.: 
payMoney); 
oracle.AddInParameter (command, ":user0", DbType.Sstring, pay.: 
payUser); 
return oracle.ExecuteNonQuery (command); 
} 
/// <summary> 
/// 得 到 新 的 支付 顺序 号 
/// </summary> 
/// <param name="audit"> 审 核 编 号 </param> 
/// <returns> 新 得 到 的 支付 顺序 号 </returns> 
/// <remarks> 
/1/ 一 次 审核 可 以 对 应 多 次 支付 ， 第 一 次 支付 序号 为 1， 第 二 次 为 2， 依 次 类 推 
/// </remarks> 
private static int newPayNo (string audit) 
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} 


string sql = " select nvl (max (Pay_ No),0) from Audit Payment where 
Audit No=:audit0"; 

Var oracle = MyDbUtility.oracle; 

Var command = oracle.GetSqlSstringCommand (sql); 
oracle.AddInParameter (command, ":audit0", DbType.String, audit); 
n = Convert.ToInt32 (oracle.ExecuteScalar (Command) ); 

return nt+l; 


int 


} 


// 从 DataReade 循环 读 取 数据 数据 生成 支付 信息 列表 
private static List<AuditPayment> fromReader (IDataReader reader) 


{ 


List<AuditPayment> list = new List<AuditPayment>(); 
while (reader.Read()) 


{ 


1 


RuditPayment p = new AuditPayment () 
P.auditNo = Convert.ToString (reader[0]); 
P.payNo = Convert .ToInt32 (reader[1]); 
P.payDate = Convert.ToDateTime (reader [2]); 
p-.payMoney = Convert.ToDouble (reader [3]); 
P.payUser = Convert.ToString (reader[4]); 
P.cancel = Convert.ToString (reader["cancel"]); 
list.Add(p); 


return list; 


(2) 在 数据 访问 层 项 目 AccountCheckDAL 中 添加 一 个 类 AuditDB, 这 个 类 实现 了 与 审 
核 相 关 的 数据 访问 功能 。 


public static class AuditDB 


有 


代码 如 下 : 


#region 数据 库 表 名 和 字段 名 称 

private const string AuditTableName = "md.bank transaction audit"; 
private const string AuditNoColumn = "audit no"; 
private const string AuditDateColumn = "audit date"; 
private const string AuditUserColumn = "audit user"; 
private const string HospitalIDColumn = "yybm"; 
private const string InsititeIDColumn = "sbjgbh"; 
private const string HospitalNameColumn = "yymc"; 
private const string AmtSumColumn = "amt sum"; 
private const string FeeSumColumn = "fee sum"; 
private const string RealAmtSumColumn = "realamt sum"; 
private const string PayMoneyColumn = "money payed"; 
private const string CancelColumn = "cancel"; 
#endregion 

// 添 加 审核 记录 


public static 


; 


/// <summary> 
/// 支付 资金 

/// </summary> 
public static int payMoney (AuditPayment payment) 


( 


} 


/// <summary> 


/// 撤销 审核 


int audit (AuditInfo audit) 


// 参 考 光盘 源 代码 


// 参 考 光 盘 源 代码 
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} 


/// </summary> 
/// <param name="audit"> 要 撤销 的 审核 ID</param> 

/// <param name="user"> 用 户 ID</param> 

public static int cancel (string audit, string user) 
{ 

2 // 参 考 光 盘 源 代码 

} 

/// <summary> 

/// 根据 条 件 得 到 审核 列表 
/// </summary> 

/// <param name="sbjgbh"> 社 保 机 构 编号 </param> 

/// <param name="dateArg"> 日 期 范围 </param> 

/// <param name="pageArg"> 分 页 参数 </param> 

public static List<AuditInfo> getAuditList (string sbjgbh,DateRangeArg 
dateArg, PageDataArgument pageArg) 

{ 

ea // 参 考 光盘 源 代码 

让 

/// <summary> 

/// 根据 审核 编号 得 到 审核 信息 
/// </summary> 

public static RuditInfo getByID(string audit) 
ps // 参 考 光 得 源 代码 

| 

/// <summary> 
/// 从 DataReader 中 循环 读 取 数 据 并 创建 审核 信息 列表 

/// </summary> 

private static List<AuditInfo> fromReader (IDataReader reader) 
{ 

a // 参 考 光盘 源 代码 

} 

/// <summary> 
/// 得 到 新 的 审核 编号 (格式 为 当前 年 月 日 6 位 +4 位 顺序 号 ， 共 10 位 字符 ) 
/// </summary> 

/// <returns></returns> 

private static string getNewID() 

a // 参 考 光 得 源 代码 

上 


(3) 业务 逻辑 层 的 主要 功能 是 调用 数据 访问 层 的 相应 方法 ， 此 处 省 略 详细 代码 。 


13.9.3 ”审核 结算 页 面 


在 图 13.25 所 示 的 结算 申请 表 〈 汇 总 格式 ) 页面 中 ， 用户 可 以 单 击 输入 一 定 结算 金额 ， 
然后 单 击 “ 审 核 确认 ”按钮 完成 审核 和 结算 支付 功能 ， 代 码 如 下 : 


protected void verify Click(object sender, EventArgs e) 
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// 生 成 审核 对 象 并 设置 各 个 属性 

AuditInfo audit2 = new AuditInfo(); 

WebUser user = Common.WebUtility.currentUser; 
audit2.amt = double.Parse (totalMoney.Text); 
audit2.auditUser = user.id; 
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audit2.fee 
CheckAccountArg arg = 
CheckAccountArg; 
Hospital h = 
audit2.yybm=arg.hospital; 
audit2.sbjgbh = arg.institute; 
audit2.yymc = h.Name; 
audit2.realamt = 
audit2.payMoney = 
AuditBLL.audit (audit2); 
audit.Enabled = false; 
audit.Visible = false; 


double.Parse (totalFee.Text); 
Session[Common.Constants.SessionCheckArg] 


as 


HospitalBLL.getByID(arg.hospital, arg.institute); 


double.Parse (realMoney.Text); 
double.Parse (payMoney .Text); 


// 添 加 审核 信息 


Page.ClientScript.RegisterStartupScript 


(this.GetType(),"auditok", 
m/seripESr)e 


13.9.4 二 次 结算 页 面 

如 前 所 述 ， 对 医疗 机 构 的 结算 可 以 一 次 性 完 
支付 外 ， 其 余 结算 支付 均 称 为 二 次 结算 。 
情况 ， 如 已 支付 金额 和 余额 ， 然 后 输入 一 
面 设计 外 观 如 图 13.27 所 示 。 


"<script>alert(" 审 核 成 功 ! 数据 已 被 记录 ! 


成 ， 也 可 以 分 成 多 次 支付 。 除 第 一 次 结算 


二 次 结算 时 ， 用 户 首先 查询 某 次 审核 对 应 的 结算 
定 的 金额 〈 不 大 于 余额 ) 进行 结 


。 二 次 结算 页 


ayAgainPage_ aspx 


结算 日 期 
Databound 
Databound 


结算 全 额 
Databound 
Databound 


Databound 
Databound 
Databound 
Databound 


Databound Databound 


Databound Databound 


Databound Databound Databound 


结算 人 
Databound 
Databound 
Databound 
Databound 
Databound 


口 SPlit | 回 Somes | 站 [Cap:Ceatentscontent2) 


Es 


图 13.27 


二 次 结算 页 面 设计 外 观 


(1) 在 表现 层 项 目 AccountCheckWeb 中 添加 一 个 用 户 控件 PaymentListControl.ascx， 
此 控件 显示 了 某 次 审核 的 所 有 支付 列表 。 控 件 代码 如 下 : 


//PaymentListControl .ascx 文件 代码 


<%@ Control Language="C#" RutoEventWireup="true" CodeBehind="PaymentList 


Control.ascx.cs" 


Inherits="AccountCheckWeb.UserControls.PaymentListControl" $%> 


<asp:GridView runat="server" ID="gridl" CssClass= 


Columns="false"> 
<Columns> 


<asp:BoundField DataField="AuditNo™" 


ItemStyle-Width="100" /> 


"gridview" RutoGenerate 


HeaderText=" 审 核 流水 号 " 
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<asp:BoundField DataField="PayNo" HeaderText=" 结 算 序 号 " ItemStyle- 
Width="80" /> 

<asp:BoundField DataField="PayDate" HeaderText=" 结 算 日 期 " 
DataFormatString="{0:D}"” ItemStyle-Width="100" /> 
<asp:BoundField DataField="PayMoney"” HeaderText=" 结 算 金额 " 
DataFormatString="{0:C}"” ItemStyle-Width="100" /> 
<asp:BoundField DataField="PayUser" HeaderText=" 结 算 人 " /> 
</Columns> 

<EmptyDataTemplate> 

<span style="color:Red;"> 没有 数据 。</span> 

</EmptyDataTemplate> 

</asp:GridView> 

//PaymentListControl.ascx.cs 文件 中 的 C# 代 码 

public partial class PaymentListControl : System.Web.UI.UserControl 
{ 


protected void Page Load(object sender, EventArgs e) 


if (needReload) 
{ 
loadData () 7 
} 
} 
// 加 载 支付 数据 


private void loadData() 

i 
if (!needReload) return; 
List<Account .Entity.RuditPayment> list = AuditPayBLL.getByAudit 
(auditNo); 
gridl.DataSource = list; 
gridl.DataBind(); 
needReload = false; 

} 

public void refresh() 

{ 
needReload = true; 

} 

// 私 有 属性 : 是 否 需要 重新 加 载 数据 


private bool needReload 


{ 
get 
{ 
string s = ViewState["needReload"] as string; 
return s == "true"; 
} 
SEE 
下 
if (value) 
ViewState["needReload"] = "true"; 
else 
ViewState["needReload"] = "false"; 
. 
} 
// 审 核 编号 


public string auditNo 
i 

det 

i 


return ViewState["theAuditNo"] as string; 
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string old = ViewState["theAuditNo"] as string; 


if (old == value) return; 
ViewState["theAuditNo"] = value; 
needReload = true; // 审 核 编号 改变 后 ， 数 据 需 要 重新 加 载 


loadData (); 


(2) 在 表现 层 项 目 AccountCheckWeb 中 添加 一 个 页 面 PayAgainPage.aspx， 页 面 代码 
如 下 : 


<asp:Content ID="Contentl" ContentPlaceHolderID="head" runat="server"> 
<script type="text/javascript"> 
// 检 测 支付 金额 是 否 合理 
function checkMoney() { 
Var remain = $('#<&s=remainLabe1.ClientIDS>') .text (); 
var pay = $('#<%=payMoney.ClientID%>') .val (); 
if (parseFloat (pay) <= 0) 
{ 
alert (' 支 付 金额 必须 大 于 零 ') ; 
return false; 
; 
if (parseFloat (pay) > parseFloat (remain)) { 
alert (' 此 次 支付 金额 不 能 大 于 余额 ! 请 修改 ! '); 
return false; 
} 
return true; 
L 
</script> 
</asp:Content> 
<asp:Content ID="Content2" ContentPlaceHolderID="ContentPlaceHolderl" 
runat="server"> 
<h3> 二 次 结算 </h3> 
审核 流水 编号 ;: <asp:TextBox ID="auditNoText" runat="server" /> 
<asp:Button ID="ok" runat="server" Text=" 查 看 " onclick="ok Click" /> 
<asp:Button ID="cancel" runat="server"” Text=" 取 消 " onclick= 
"eantel Click™ /> 
<asp:Panel runat="server" ID="payPanel"> 
此 次 结算 金额 : <asp:TextBox runat="server" ID="payMoney" /> 
<asp:Button runat="server" ID="payButton" Text=" 结 算 " OnClientClick= 
"return checkMoney();" onclick="payButton Click" /><br /> 
<br /> 此 次 审核 详细 信息 <br /> 
<table> <tr> 
<td> 审 核 日 期 </td><td><asp:Label runat="server" ID="auditDateLabel" 
/></td> 
<td> 审 核 人 </td><td><asp:Label runat="server" ID="auditUserLable" 
/></td> 
<td> 应 付 金 额 </td><td><asp:Label runat="server" ID="paylLabel" 
/></td> 
<td> 已 付 金 额 </td><td><asp:Label runat="server" ID="pay2Label" 
/></td> 
<td> 余 额 </td><td><asp:Label runat="server" ID="remainLabel" /></td> 
</tr></table> 


0 


发 
吕 
了 
将 
识 
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<br /> 
此 次 审核 已 结算 情况 列表 <br /> 
<ucl:PaymentListControl ID="payList1"” runat="server"/> 
</asp:Panel> 
</asp:Content> 


(3) 在 PayAgainPage.aspx 页 面 中 “查看 ”按钮 的 Click 事件 中 ， 根 据 用 户 输入 的 审核 
编号 取出 审核 信息 及 支付 列表 ， 并 显示 在 页 面 上 。 


protected void ok Click(object sender, EventArgs e) 


payMode (false) 
AuditInfo audit=AuditBLL.getByID (auditNoText -Text) // 查 找 审核 信息 
if (audit == null) return; 
if (audit.moneyRemained <= 0) 
{ 
ClientScript.RegisterStartupScript (this.GetType(),"NoRemain", 
"<script>alert (' 此 次 审核 已 经 结算 完毕 , 不 能 再 次 结算 ! ') ;</script>"); 
return; 
} 
payMode (true); // 进 入 支付 模式 
// 将 得 到 的 审核 信息 显示 在 各 个 控件 中 
auditUserLable.Text = audit.auditUser; 
auditDateLabel .Text = audit.auditDate.ToShortDateString(); 
paylLabel.Text = audit.realamt.ToString("0.00"); 
pay2Label.Text = audit.payMoney.ToString("0.00"); 
remainLabel .Text = audit.moneyRemained.ToString("0.00"); 
payListl.auditNo = auditNoText .Text; 


} 
// 是 否 进入 支付 模式 不同 模 式 下 各 个 控件 可 见 性 、 可 用 性 不 同 》 
private void payMode (bool flag) 
{ 
payPanel .Visible = flag; 
payMoney .Readonly !flag; 
payButton.Enabled = flag; 
auditNoText .Readonly = flag; 


} 
(4) 在 PayAgainPage.aspx 页 面 “ 支 付 ” 按 钮 的 Click 事件 中 ， 完 成 支付 功能 。 


protected void payButton Click(object sender, EventArgs e) 
{ 
AuditPayment pay = new RuditPayment (); 
pay.auditNo = auditNoText.Text; 
pay.payMoney = double.Parse (payMoney.Text); 
pay.payUser = WebUtility.currentUser.id; 
AuditBLL.payMoney (pay); 
ClientScript.RegisterStartupScript (this.GetType(), 
"successpay"，"<script>alert (' 结 算 成 功 ! ');</script>"); 
payMode (false); 
payListl1.refresh(); // 刷 新 支付 列表 
} 


(5) 运行 PayAgainPage.aspx 页 面 ， 运 行 界面 如 图 13.28 所 示 。 
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过 
用 户 权限 管理 


市 核 注水 坊 号 [1005040000 查看 取消 
| 此 次 结算 全 凑 。 |500 结算 


此 次 审核 详细 信息 
审核 日 期 2010-5-4 审核 人 adnin 应 讨 金 额 495. 50 已 付 全 额 !1936. 50 余 颜 1500. 00 


此 次 审核 已 结算 情况 列表 
EECEEIIDI 结算 全 额 。 结算 人 


1005040000 上 2010 年 5 月 4 日 。 Yl,096.50 adain 
1005M0000 2 2010 和 5 月 4 日 。 等 400. 00 adain 


2010 年 3 有 日。 入 500. 00 


1005040000 


图 13.28 二 次 结算 页 面 运行 界面 


13.10 统计 报表 


根据 用 户 需求 ， 用 户 可 以 对 社保 卡 结算 数据 进行 统计 生成 报表 。 用 户 要 求 产生 5 种 统 
计 报 表 ， 由 于 有 些 报 表 实 现 方式 类 似 ， 受 篇 幅 限制 ， 本 节 只 介绍 其 中 两 种 报表 的 设计 和 
实现 。 


13.10.1 审核 结算 明细 表 


在 审核 结算 明细 表 中 ， 用 户 根据 时 间 范 围 查 询 审核 结算 明细 ， 并 可 以 撤销 审核 。 审 核 
结算 明细 表 界面 如 图 13.29 所 示 。 


一 人 
划 NE 系 绞 om EM necon 


{ 


社保 机 榴 : [371301 司 9m: [201c-05-01 医 2010-05-30 E32 


eels 沪 厅 二。 市 校 日期 市 校 人 医院 所 到 医 所 名 称 消 玉 全 祝 手 疾 入 应 结算 全 颜 已 结算 全 额 拉 销 标志 详情 披 消 
lots 所 Aa" 


1005040000 2010-05-04 aaain yy3  y3 3514.20 17.70 3496.50 


图 13.29 审核 结算 明细 表 
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(1) 在 表现 层 项 目 AccountCheckWeb 中 添加 一 个 嵌 套 母 版 页 ReportMastermaster， 此 
母 版 页 在 Nrcm.master 母 版 页 的 基础 上 添加 了 左 侧 导航 栏 。 此 母 版 页 是 所 有 报表 页 面 的 母 
版 。ReportMastermaster 页 面 代码 如 下 : 


<asp:Content ID="Content2" ContentPlaceHolderID="ContentPlaceHolder1l" 
runat="server"> 

<div id="leftNav"> 
<a href="ConsumeReportPage.aspx"> 消 费 汇 总 统计 </a><br /><hr /> 
<a href="CompensationReportPage.aspx"> 结 算 汇总 统计 </a><br /><hr /> 
<a href="CompensationStatePage.aspx"> 结 算 情况 查询 </a><br /><hr /> 
<a href="AuditCompensationPage.aspx"> 审 核 情 况 查 询 </a><br /><hr /> 
<a href="PayReportPage .aspx"> 分 批 结算 查询 </a><br /><hr /> 
</div> 
<div id="rightContent"> 
<asp:ContentPlaceHolder ID="ContentPlaceHolder1l"” runat="server"> 
</asp:ContentPlaceHolder> 
</div> 
</asp:Content> 


(2) 在 表现 层 项 目 AccountCheckWeb 中 ， 以 ReportMastermaster 为 母 版 页 添加 一 个 新 
页 面 AuditCompensationPage.aspx， 页 面 布局 如 图 13.29 所 示 ， 页 面 代 码 如 下 : 


<asp:Content ID="Content2" ContentPlaceHolderID="ContentPlaceHolder1l" 
runat="server"> 
<h3> 审 核 结算 明细 表 </h3> 社保 机 构 : 
<asp:DropDownList ID="hospitalList" runat="server"></asp: 
DropDownList> 
日 期 : <asp:TextBox ID="datel" runat="server"></asp:TextBox> 
至 <asp:TextBox ID="date2" runat="server"></asp:TextBox>&nbsp; &nbsp; 
<asp:Button ID="OK" runat="server" Text=" 查 询 " onclick="OK Click" /> <br 
/> 
<div> 
<asp:GridView ID="gridl" runat="server" CssClass="gridview" 
AutoGenerateColumns="false" 
onselectedindexchanging="gridl SelectedIndexChanging" Data 
KeyNames="AuditNo"> 
<Columns> 
<asp:BoundField DataField="AuditNo"” HeaderText=" 流 水 号 " /> 
<asp:BoundField DataField="AuditDate" HeaderText=" 审 核 日 期 " Data 
FormatSstring="{0:yyyy-MM-dd}" /> 
<asp:BoundField DataField="AuditUser" HeaderText=" 审 核 人 " /> 
<asp:BoundField DataField="yybm" HeaderText=" 医 院 编码 ” /> 
<asp:BoundField DataField="yymc"” HeaderText=" 医 院 名 称 ” /> 
<asp:BoundField DataField="amt" HeaderText=" 消 费 金额 " 
DataFormatString="{0:0.00}"” /> 
<asp:BoundField DataField="fee" HeaderText=" 手 续费 " DataFormatString= 
D000P 
<asp:BoundField DataField="realamt" HeaderText=" 应 结算 金额 " 
DataFormatString="{0:0.00}"/> 
<asp:BoundField DataField="payMoney"” HeaderText=" 已 结算 金额 " 
DataFormatstring="{0:0.00}"/> 
<asp:TemplateField HeaderText=" 撤 销 标志 "> 
<ItemTemplate> 
<asp:CheckBox runat="server" ID="cancelFlag" 
Checked=" <%#Eval ("cancel") .ToString() .Trim()=="1"%>' Enabled= 
"false" /> 
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</ItemTemplate> 

</asp: TemplateField> 

<asp:TemplateField HeaderText=" 详 情 "> 

<ItemTemplate> 

<a href="../report/PayReportPage.aspx?audit=<%#Eval ("AuditNo")$>"> 
<img src="../images/view.png”alt=" 查 看 " style="border:none;" /></a> 
</ItemTemplate> 

</asp:TemplateField> 

<asp:CommandField ShowSelectButton="true" SelectImageUTr1= 
"../images/undo.png"” ButtonType="Image" HeaderText=" 撤 销 "” /> 


</Columns> 
<EmptyDataTemplate> 
<span style="color:Red; font-weight:bold; "> 没有 符合 条 件 的 数据 
</span> 
</EmptyDataTemplate> 


</asp:GridView> 
<webdiyer:AspNetPager ID="pagerl" runat="server"> 
</webdiyer:AspNetPager> 

</div> 

</asp:Content> 


(3) 在 AuditCompensationPage.aspx 页 面 的 Page_Load 事件 中 ， 绑 定 所 有 社保 机 构 列 
表 以 供用 户 选择 。 


protected void Page Load (object sender，EventRrgs e) 


{ 


if (!IsPostBack) 
{ 
bindsbjgList(); 
} 
// 绑 定 社保 机 构 列表 
private void bindsbjgList() 
{ 
List<string> list = SbjgBLL.getAll(); 
foreach (string item in list) 
{ 
hospitalList.Items.Add (item); 
} 


(4) 在 AuditCompensationPage.aspx 页 面 “ 查 询 ” 按 钮 的 Click 事件 中 ,根据 用 户 指定 
的 条 件 查询 数据 并 显示 。 
// 绑 定 审核 结算 列表 


private void bindList() 
{ 
/ /创建 分 页 参数 
PageDataArgument page = new PageDataArgument () { refreshCount = true }; 
page.pageIndex = pagerl.CurrentPageIndex - 1; 
page.pageSize = pagerl .PageSize; 
// 得 到 审核 数据 列表 ， 并 绑 定 到 GridView 
List<AuditInfo> list = AuditBLL.getAuditList (hospitalList. 
SelectedValue, 
DateRangeArg .between2Date (datel .Text, date2.Text),page); 
pagerl .RecordCount = page.count; 
gridl.DataSource = list; 
gridl.DataBind(); 
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// 单 击 查询 按钮 时 ， 根 据 查 询 条 件 显 示 第 1 页 数据 
protected void OK _ Click(object sender, EventArgs e) 
{ 
pagerl.CurrentPageIndex = 1; 
bindList(); 
S 


(5) 在 AuditCompensationPage.aspx 页 面 GridView 的 SelectedIndexChanged 事件 中 ， 
撤销 GridView 当前 选择 的 审核 。 


protected void gridl SelectedIndexChanging (object sender, 
GridViewSelectEventArgs e) 
| 
e.Cancel = true; // 取 消 当 前 事件 
// 得 到 CheckBox 控件 
CheckBox check = gridl.Rows [e.NewSelectedIndex] .FindControl 
("cancelFlag") as CheckBox; 
// 如 果 当 前 选中 记录 已 经 被 撤销 则 返回 
if (check.Checked) 
return; 
// 得 到 审核 编号 并 撤销 此 次 审核 
string audit = gridl.DataKeys [e.NewSelectedIndex] [0] .ToString(); 
AuditBLL.cancel (audit, AccountCheckWeb.Common.WebUtility. 
currentUser.id); 
WebUtility.registerStartupScript (this，"alert (' 已 经 撤销 此 次 审核 ! ') ;"); 
bindList(); 


13.10.2 ”结算 情况 统计 表 


在 结算 情况 统计 表 中 ， 可 以 查询 某 社保 机 构 已 经 结算 和 尚未 结算 的 社保 消费 数据 ， 运 
行 界面 如 图 13.30 所 示 。 


加 市 动 局 社保 不 结 复 对 沪 系 统一 esills Tirefwz 
文件 加。 柏 煤 尼 ) 考 看 呈 。 后 史 名 书 革 工具 加 和 胞 人 


[<n [re 
EL FERTA-T 人 EI LE 
而 芝 动 局 社保 雪 星 对 素 统 
市 劳动 局 社保 卡 结算 对 账 系统 ng, sya. 
E 宇 < 包 a Ln me 
医保 账 日 核对 “银行 数据 上 传 ”数据 查询 统 计 “ 响 用 二 次 结算 ” 阮 疗 机 构 字典 ”用 户 权限 管理 
请 区 C9 信和 定点 机 构 结 算 情 况 统计 


ET SEE 


当前 期 ，2010 下 5 月 4 Tuczday 


绩 复 江 站 统计 
一 社 但 机 榴 : 2 eit- 居间 所 宁 续 算 。 统 计 
de ES 医院 坟 记 医院 名 称 谓 嫩 全 六 手续 费 车 站 全 共 
刷 术 情况 查 向 36 言 守门 疹 于 2 110.40 ¥10.96 ¥2, 159, 44 
分 批 结算 查询 yy 仁 言 中 醒 匡 门诊 ¥a42.80 ¥q.28 于 838.52 
一 六 镇 相 李 其 乔 字 卫生 主 ¥38.20 ¥0.20 ¥38.00 

yyT 山东 流 北 实业 公司 ¥5, 949.20 ¥58. 14 于 889. 46 


图 13.30 ”结算 情况 统计 表 


(1) 在 数据 访问 层 项 目 AccountCheckDAL 中 添加 一 个 类 ReportDB， 用 以 产生 各 种 报 
表 数 据 。 代 码 如 下 : 


public static class ReportDB 
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/ /按照 医疗 机 构 进行 消费 汇总 统计 
public static List<HospitalReport> consumeReport (string sbjgbh, 
DateRangeArg date) 


} 


return consumeReportByAudit (sbjgbh, date, '*'); 


// 按 钮 医疗 机 构 进行 消费 汇总 《只 统计 未 审核 数据 ) 
public static List<HospitalReport> unauditReport (string sbjgbh, 
DateRangeArg date) 


下 
} 


return consumeReportByAudit (sbjgbh, date, '0'); 


/// <summary> 

/// 根据 审核 结算 状态 查询 消费 汇总 

/// </summary> 

/// <param name="sbjgbh"> 社 保 机 构 编号 </param> 

/// <param name="date"> 日 期 范围 </param> 

/// <param name="auditState"> 审 核 状 态 ， 必 须 为 * 或 0，* 表 示 所 有 数据 ，0 表示 未 审 


核 数据 </param> 


private static List<HospitalReport> consumeReportByAudit (string 
sbjgbh, DateRangeArg date, char auditState) 


if (!(auditState == '*' || auditState == '0')) 
throw new ArgumentException(" must be * or 0 ","auditSstate"); 


// 生 成 where 条 件 
string predicate = 
if (auditstate 外 
predicate = " and (audit no is null) "7 
else if (auditstate 让 | 
predicate = " "7 
// 构 建 select 语句 
string sqll = "select sum(a.amt) as amt,sum(a.fee) as fee, " 
+"Ssum(a.realarmt) as realamt,b.yybm from " 
+" (select sum(amt) as amt,sum(fee) as fee ,sum(realamt) as 
realamt, corno,term " 
+" from bank transaction where " 
+" (time between :datel and :date2) "+predicate 
+" group by corno,term) a " 
+" inner join institute dictionary b " 
+" on a.corno=b.corno and a.term=b.term and b.sbjgbh=:sbjgbh " 
+" group by b.yybm "7 
var oracle = MyDbUtility.oracle; 
/ /创建 命 令 对 象 ， 并 添加 各 个 参数 
DbCommand command = oracle.GetSqlStringCommand(sql1) ; 
oracle.AddInParameter (command, ":sbjgbh", DbType.String, sbjgbh); 
oracle.AddInParameter (command, ":datel", DbType.DateTime, 
date.from); 
oracle.AddInParameter (command, ":date2", DbType.DateTime, 
date.to); 
// 执 行 查 询 ， 得 到 数据 读 取 器 ， 读 取 数 据 
IDataReader reader = oracle.ExecuteReader (command) ; // 执 行 查询 命令 
List<HospitalReport> list = new List<HospitalReport>(); 
while (reader.Read()) 
{ 


// 生 成 报表 数据 

HospitalReport item = new HospitalReport (); 
item.amt = Convert .ToDouble (reader ["amt"]) > 
item.fee = Convert .ToDouble (reader["fee"]); 
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} 


} 


} 


item.realamt = Convert.ToDouble (reader["realamt"]); 
item.yybm = Convert.ToString (reader["yybm"]); 
item.sbjgbh = sbjgbh; 
string yymc = "没有 对 应 医院 "; 
if (!(string.IsNullOrEmpty(item.yybm) || string. 
IsNullOrEmpty (item.sbjgbh) )) 
是 
var h = HospitalDB.getByID (item.yybm, item.sbjgbh); 
if (h != null) 
yymc = h.Name; 
,4 
item.yymc = yymc; 
list.Add (item); 


reader.Close(); 
command.Dispose(); 
return list; 


// 审 核 结 算 汇 总 统计 
public static List<HospitalReport> auditReport (string sbjgbh, 
DateRangeArg date) 


. 


// 参 考 光 税源 代码 


(2) 在 业务 逻辑 层 项 目 AccountCheckBLL 中 添加 一 个 ReportBLL 类 ， 代 码 如 下 : 


public static class ReportBLL 


{ 


{ 


/// <summary> 

/// 根据 社保 机 构 编 号 、 日 期 范围 和 报表 种 类 生成 报表 数据 

/// </summary> 

public static List<HospitalReport> getReport (string sbjgbh, 
DateRangeArg date,ReportType type) 


{ 


} 


switch (type) 


1 


} 


case ReportType.ConsumeReport: 
return ReportDB.consumeReport (sbjgbh, date); 
case ReportType.UnauditReport: 
return ReportDB.unauditReport (sbjgbh, date); 
case ReportType.AuditReport: 
return ReportDB.auditReport (sbjgbh, date); 
default: 
throw new ApplicationException(" 不 能 识别 的 报表 类 型 ") ; 


3 

// 报 表 种 类 枚 举 类 型 

public enum ReportType 
ConsumeReport, /7 消费 报表 
UnauditReport, // 未 审核 报表 
AuditReport // 已 审核 报表 


(3 ) 在 表现 层 项 目 AccountCheckWeb 中 添加 一 个 页 面 CompensationStatePage.aspx， 页 
面 外 观 如 图 13.30 所 示 ， 页 面 代码 如 下 : 
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<asp:Content ID="Content2" ContentPlaceHolderID="ContentPlaceHolder1" 
runat="server"> 


<h3> 定 点 机 构 结算 情况 统计 </h3> 

社保 机 构 : <!-- 社 保 机 构 列 表 --> 
<asp:DropDownList ID="hospitalList" runat="server"> 
</asp:DropDownList> 

日 期 : <asp:TextBox ID="datel" runat="server"></asp:TextBox> 

至 <asp:TextBox ID="date2" runat="server"></asp:TextBox> 
<asp:RadioButton ID="auditRadio" runat="server" GroupName="groupl" 
Text=" 已 结算 "Checked="true"/> 

<asp:RadioButton ID="unauditRadio" runat="server" GroupName="groupl" 
Text=" 示 结算" />gnbsp; gnbsp; 

<asp:Button ID="OK" runat="server" Text=" 统 计 " OnClientClick="return 
checkInput ();" onclick="OK Click" /> <br /> 

<div> 

<asp:GridView ID="gridl" runat="server" CssClass="gridview" 
AutoGenerateColumns="false"> 

<Columns> 

<asp:BoundField DataField="yybm"” HeaderText=" 医 院 编码 " 
ItemStyle-Width="100" /> 

<asp:BoundField DataField="yymc"” HeaderText=" 医 院 名 称 " 
ItemStyle-Width="200"/> 

<asp:BoundField DataField="amt" HeaderText=" 消 费 金 额 " 
DataFormatString="{0:C}" ItemStyle-Width="100"/> 

<asp:BoundField DataField="fee" HeaderText=" 手 续费 " 
DataFormatString="{0:C}" ItemStyle-Width="100"/> 

<asp:BoundField DataField="realamt" HeaderText=" 结 算 金额 " 
DataFormatString="{0:C}" ItemStyle-Width="100"/> 


</Columns> 
<EmptyDataTemplate> <!-- 没 有 数据 时 显示 的 提示 信息 --> 
<span style="color:Red; font-weight:bold;"> 没 有 符合 条 件 的 数据 
</span> 
</EmptyDataTemplate> 


</asp:GridView> 


</div> 
</asp:Content> 


(4) 在 CompensationStatePage.aspx 页 面 “统计 ”按钮 的 Click 事件 中 ， 根 据 用 户 输入 
的 条 件 查询 数据 ， 并 显示 在 GridView 中 。 


protected void OK Click(object sender, EventArgs e) 


{ 


List<HospitalReport> list=null; 


// 得 到 所 查询 的 日 期 范围 
DateRangeArg date=DateRangeArg .between2Date (datel]l.Text, date2.Text); 
string sbjg = hospitalList.SelectedValue; /7 社保 机 构 编号 


// 根 据 单 选 按钮 ， 进 行 不 同 查询 
if (auditRadio.Checked) 

list = ReportBLL.getReport (sbjg,date , ReportType.AuditReport); 
else 

list = ReportBLL.getReport (sbjg, date, ReportType.UnauditReport); 
// 将 查询 结果 绑 定 到 GriqView 控件 
gridl.DataSource = list; 
gridl.DataBind(); 
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13.11 结 


本 章 介绍 了 作者 为 A 市 人 力 资源 和 社会 保障 局 开发 的 一 个 软件 项 目 : 社 保 卡 结算 系统 。 
本 项 目 使 用 Oracle 作为 后 台数 据 库 ， 数 据 关 系 较为 复杂 ， 应 用 程序 功能 主要 包括 银行 数据 
导入 、 费 用 审核 、 资 金 结算 、 统 计 报 表 等 功能 。 
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新 型 农村 合作 医疗 〈 简 称 新 农 合 ) 作为 造福 广大 农民 的 一 项 重大 举措 ， 受 到 国家 和 各 
级 政府 的 高 度 重 视 。 卫 生 部 要 求 各 地 建立 起 与 新 农 合 制度 发 展 相 适应 、 与 建设 中 的 国家 卫 
生 信息 系统 相 衔接 、 较 为 完备 和 高 效 的 全 国 新 农 合 信息 系统 。 在 各 级 新 农 合 管理 部 门 、 经 
办 机 构 、 定 点 医疗 机 构 及 其 他 相关 部 门 间 建 立 计算 机 网 络 联接 ， 实 现 网 上 在 线 审核 结算 、 
实时 监控 和 信息 汇总 ， 实 现 新 农 合 业务 管理 的 数字 化 、 信 息 化 、 科 学 化 ， 提 高 新 农 合 工作 
效率 和 服务 水 平 。 

在 这 种 形势 下 ， 各 级 新 型 农村 合作 医疗 管理 部 门 都 开始 了 信息 化 建设 工作 。 各 软件 公 
司 也 抓 住 这 一 机 遇 ， 开 发 了 各 种 新 型 农村 合作 医疗 管理 信息 系统 ， 简 称 新 农 合 系统 。 本 章 
将 讲解 作者 所 开发 的 一 套 新 农 合 系统 。 


14.1 整体 设计 思 


新 农 合 系统 用 户 数量 多 ， 地 域 履 盖 范围 广 ， 功 能 模块 复杂 。 在 开发 此 系统 以 前 ， 需 要 
详细 了 解 新 农 合 系统 工作 流程 和 用 户 需求 。 本 节 将 介绍 新 农 合 系统 要 实现 的 主要 功能 及 系 
统 的 整体 结构 。 


14.1.1 新 农 合 业 务 流 程 


新 型 农村 合作 医疗 ， 简 单 来 说 就 是 给 农村 实行 的 医疗 保险 。 保 险 资金 的 筹集 包括 3 部 
分 : 省 财政 拨款 、 县 财政 拨款 和 农民 个 人 缴费 。 例 如 ， 每 个 农民 参加 新 农 合 ， 每 年 需要 缴 
纳 20 元 ， 县 级 财政 拨款 50 元 ， 省 级 财政 拨款 50 元 ， 投 保 资 金 一 共 是 120 元 。 所 有 农民 的 
投保 资金 汇总 到 一 起 ， 就 形成 了 新 农 合 基金 。 当 参合 农民 生病 住院 时 ， 就 可 以 按照 相关 政 
策 报销 一 定 比例 的 费用 ， 而 这 些 报销 费用 从 新 农 合 基金 中 支取 。 

在 新 农 合 刚 开始 实施 阶段 ， 还 没有 实现 信息 化 ， 农 民 住院 后 ， 需 要 拿 着 相关 证 明 (如 
住院 费用 发 票 、 病 历 复印 件 等 ) 到 合作 医疗 管理 办 公 室 进行 审核 和 报销 。 这 种 报销 方式 周 
期 长 ， 成 本 高 ， 效 率 低 。 为 了 使 新 农 合 政策 的 实施 更 加 顺畅 ， 各 地 都 先后 对 新 农 合 业务 实 
行 了 信息 化 管理 。 

实行 信息 化 管理 以 后 ， 新 农 合 系统 实现 全 市 甚至 全 省 联网 ， 各 种 数据 〈 如 病人 信息 、 
药品 信息 、 医 生 信 息 等 ) 都 可 以 实现 共享 。 农 民 出 院 结算 时 ， 根 据 身 份 证 号 ， 即 可 得 到 农 
民 的 参合 信息 (如 是 否 参 合 、 本 年 度 可 报销 金额 等 ), 新 农 合 系统 会 自动 计算 报销 金额 并 
从 住院 费用 中 扣除 报销 金额 ， 农 民 只 需 支付 其 余 的 金额 给 医院 。 例 如 ， 某 参合 农民 住院 共 
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花费 3000 元 ， 可 报销 1000 元 ， 则 此 病人 只 需 交 给 医院 2000 元 ， 其 余 1000 元 从 新 农 合 基 
金 中 支付 。 

参合 农民 出 院 时 ， 实 际 缴纳 的 出 院 费 用 低 于 农民 的 真实 费用 ， 其 差额 由 新 农 合 基金 支 
付 。 这 些 资金 需要 医院 到 新 农 合 办 公 室 领 取 。 每 隔 一 段 时 间 ， 医 院 需要 到 新 农 合 办 公 室 进 
行 结算 。 此 时 新 农 合 系统 会 计算 该 时 间 段 内 共有 多 少 参合 农民 在 此 医院 住院 ， 花 费 了 多 少 
钱 ， 应 该 由 新 农 合 报销 多 少 钱 ， 然 后 将 报销 金额 支付 给 医院 。 至 此 ， 医 院 收 到 了 病人 的 全 
部 住院 费用 。 根 据 以 上 分 析 ， 可 以 得 到 新 农 合 系统 业务 流程 图 如 图 14.1 所 示 。 


图 14.1 新 农 合 业务 流程 图 


14.1.2 系统 功能 模块 


新 农 合 系统 的 用 户主 要 包括 两 类 : 一 类 是 各 级 新 农 合 管理 部 门 ， 如 市 、 县 〈 区 ) 卫生 
局 ， 乡 (镇 ) 合作 医疗 管理 办 公 室 ， 另 一 类 是 定点 医疗 机 构 ， 即 新 农 合 定点 医院 。 新 农 合 
管理 部 门 使 用 新 农 合 系统 进行 基础 数据 管理 、 费 用 审核 等 操作 ， 定 点 医疗 机 构 使 用 新 农 合 
系统 进行 住院 费用 结算 、 药 品 字 典 管理 等 操作 。 整 个 系统 功能 如 图 14.2 所 示 。 


1. 数据 字典 管理 

此 模块 用 于 维护 系统 中 用 到 的 基础 数据 ， 如 职业 、 民 族 、 行 政 区 划 等 。 

2. 参合 档案 管理 

此 模块 用 于 维护 参合 农民 和 家 庭 信息 。 

3. 参合 缴费 

此 模块 实现 参合 农民 缴费 和 单据 打印 功能 。 

4. 药品 目录 管理 

此 模块 用 于 管理 基本 用 药 目 录 ， 只 有 在 目录 范围 之 内 的 药品 才能 报销 ， 而 且 不 同 药品 
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有 不 同 的 报销 比例 。 
5. 报销 政策 管理 
此 模块 用 于 设置 报销 政策 ， 如 起 报 线 、 分 段 报 销 比例 、 单 病 种 报销 金额 等 。 


报销 政策 管理 
间 病人 费用 审核 
合 
入 
理 
间 补偿 结算 
新 i 
统计 报表 
鸯 药品 目录 管理 
统 
住院 结算 
药品 对 应 
统计 报表 
图 14.2 新 农 合 系统 功能 模块 图 
6. 病人 费用 审核 


此 模块 用 于 审核 病人 住院 费用 ， 检 查 是 否 存在 诸如 不 合理 使 用 昂贵 药品 、 虚 假 住院 费 
用 等 情况 。 

7. 补偿 结算 

根据 病人 在 医院 的 住院 费用 与 医院 进行 补偿 结算 。 

8. 统计 报表 

此 模块 用 于 生成 各 种 统计 报表 。 

9. 住院 结算 


参合 病人 出 院 时 ， 医 疗 机 构 需 要 将 数据 上 传 到 新 农 合 数据 库 中 ， 并 计算 病人 需要 实际 
支付 的 住院 费用 金额 和 新 农 合 报销 金额 。 


10. 药品 对 应 
于 各 个 医院 药品 编码 不 一 致 ， 各 医院 需要 将 各 自 的 药品 对 应 到 新 农 合 系统 中 的 相应 


-人 


药品 ， 以 实现 数据 传输 和 共享 。 
14.1.3 ”数据 库 结构 


新 农 合 系统 数据 结构 较为 复杂 ， 数 据 库 中 共有 大 约 50 个 表 ， 如 图 14.3 所 示 。 在 了 解 
新 农 合 系统 的 细节 功能 以 前 ， 不 容易 理解 数据 库 的 结构 ， 此 处 不 详细 介绍 数据 库 结构 ， 等 
到 实现 某 个 功能 时 再 具体 介绍 相关 表 结构 。 


XH RD NEV IAG EW NEC Hm 
和 ta 内 | 夸 古 忆 /| 所 同 多 /从 日 除 太 也 癌 
EE 3 


图 14.3 新 农 合 数据 库 中 的 表 


14.1.4 搭建 项 目 框架 


新 农 合 系统 采用 多 层 结构 设计 ， 使 用 实体 框架 实现 数据 访问 功能 。 整 个 新 农 合 项 目 解 
决 方案 中 包含 以 下 几 个 项 目 : 实体 框架 层 、 数 据 访问 层 、 业 务 轴 辑 层 、Web 表现 层 、 公 共 
类 库 、 单 元 测试 项 目 、 短 信 通 知 。 搭 建 整个 解决 方案 框架 的 步骤 如 下 。 

(1) 创建 一 个 空白 解决 方案 NRCM， 由 于 解决 方案 中 项 目 较 多 ， 为 了 使 项 目 结构 更 加 
清晰 ， 在 解决 方案 中 添加 两 个 解决 方案 文件 夹 : DLL 和 Test， 分 别 用 于 包含 类 库 项 目 和 测 
试 项 目 。 

(2) 在 解决 方案 DLL 文件 夹 中 ， 添 加 一 个 类 库 项 目 Nrcm.Entity 作为 实体 框架 层 。 

(3) 在 解决 方案 DLL 文件 夹 中 ， 添 加 一 个 类 库 项 目 Nrem.Dal 作为 数据 访问 层 。 

(4) 在 解决 方案 DLL 文件 夹 中 ， 添 加 一 个 类 库 项 目 Nrcm.Bll 作为 业务 逻辑 层 。 

(5) 在 解决 方案 DLL 文件 夹 中 ， 添 加 一 个 类 库 项 目 Nrem.Common， 此 项 目 中 包含 其 
他 各 个 项 目 都 会 用 到 的 公共 类 。 

(6) 在 解决 方案 中 添加 一 个 ASPNET Web 应 用 程序 项 目 Nrem.Web， 作 为 表现 层 。 

(7) 在 解决 方案 Test 文件 夹 中 添加 一 个 类 库 项 目 DalTest， 作 为 数据 访问 层 的 单元 测 
试 项 目 。 

(8) 在 解决 方案 Test 文件 夹 中 添加 一 个 类 库 项 目 BllTest,， 作 为 业务 逻辑 层 的 单元 测试 
项 目 。 

(9) 在 各 个 项 目 之 间 添 加 引用 关系 ， 表 现 层 Nrcm.Web、 业 务 逻辑 层 Nrem.Bll、 数 据 
访问 层 Nrem.Dal 都 引用 实体 框架 层 Nrem.Entity 和 公共 类 库 Nrcm.Common， 业 务 罗 辑 层 
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Nrcm.Bll 引用 数据 访问 层 Nrem.Dal， 表 现 层 Nrem.Web 引用 业务 逻辑 层 Nrem.Bll， 测 试 项 
目 引 用 被 测试 项 目 。 整 个 解决 方案 结构 及 各 个 项 目 之 间 的 关系 如 图 14.4 所 示 。 


Web 表 现 层 


Nrcm.Web 
业务 逻辑 层 测试 项 目 
Nrem.BIl BLL BllTest 
数据 访问 层 DAL 测 试 项 目 
Nrem.Dal DalTest 
实体 框架 层 通用 类 库 
Nrem.Entity Nrem.Common 


图 14.4 新 农 合 系统 项 目 结构 


14.2 ” 母 版 页 设计 


新 农 合 系统 的 页 面 基本 由 4 部 分 组 成 : 页 面 项 部 的 LOGO 和 工具 栏 ， 页 面 左 侧 的 导航 
栏 ， 页 面 右 侧 的 页 面 正文 ， 页 面 底部 的 页 脚 。 由 于 新 农 合 系统 功能 模块 较 多 ， 为 了 条 理 清 
楚 地 组 织 所 有 模块 ， 系 统 使 用 页 面 顶部 的 工具 栏 和 页 面 左 侧 的 导航 栏 进行 功能 导航 。 页 面 
顶部 的 工具 栏 作为 第 一 级 菜单 ， 页 面 左 侧 导航 栏 作为 第 二 级 菜单 。 一 个 典型 的 新 农 合 系统 
的 页 面 结构 如 图 14.5 所 示 。 


We 2 De re er 
rm EY 


地 凋 全 医疗 机 档 
省 级 及 以 上 医疗 机 村 
其 他 于 这 机 构 


日 
RN 


图 14.5 新 农 合 系统 典型 页 面 结构 


和 
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在 图 14.5 所 示 的 新 农 合 页 面 中 ， 项 部 的 页 头 和 底部 的 页 脚 对 于 所 有 页 面 都 是 相同 的 ， 
因此 可 以 将 其 放 于 母 版 页 中 。 虽 然 大 多 数 页 面 左 侧 都 有 导航 栏 ， 但 是 也 存在 个 别 没 有 左 侧 
导航 栏 的 页 面 , 而 且 对 于 不 同 的 一 级 菜单 (工具 栏 上 的 按钮 ), 左 侧 导 航 栏 的 内 容 也 不 相同 。 
为 了 同时 兼顾 通用 和 定制 ， 本 系统 采用 两 级 母 版 页 ， 即 主 母 版 页 中 仅 包含 页 头 页 脚 内 容 ， 
对 于 不 同 的 功能 模块 (如 报销 政策 管理 模块 )， 再 基于 主 母 版 页 创建 一 个 二 级 母 版 页 ， 作 为 
该 模块 中 所 有 页 面 的 母 版 页 。 


14.2.1 天气 预报 用 户 控件 


在 新 农 合 系统 母 版 页 的 顶部 显示 了 今天 的 天 气 情况 ， 该 数据 来 源 于 互联 网 上 发 布 的 天 
气 预报 Web 服务 。 本 系统 使 用 一 个 用 户 控件 Weatherascx 封装 了 天 气 预报 功能 。 创 建 
Weatherascx 用 户 控件 的 步骤 如 下 。 

(1) 在 表现 层 项 目 Nrcm.Web 中 添加 一 个 Web 服务 引用 , 地 址 为 http://www.ayandy.com 
/Service.asmx， 此 Web 服务 能 够 免费 提供 全 国 各 地 的 天 气 预 报 数据 。 

(2) 在 表现 层 项 目 Nrcm.Web 中 添加 一 个 类 WeatherServcie， 封 装 对 天 气 预 报 Web 服 
务 的 调用 。 由 于 天 气 预 报 是 一 种 相对 稳定 的 数据 ， 没 有 必要 频繁 调用 Web Service 浪费 性 
能 ， 本 系统 每 隔 10 分 钟 刷新 一 次 天 气 预 报 数据 。 在 代码 中 判断 当前 时 间距 上 次 调用 Web 
Service 刷新 天 气 信息 是 否 已 达到 或 超过 10 分 钟 ， 如 果 未 到 10 分 钟 则 不 调用 Web Service， 
而 是 使 用 上 次 得 到 的 天 气 数据 。 

internal class WeatherService 


上 


internal const string city = "北京 "; 


internal static DateTime lastGetTime; // 上 次 获得 天 气 信息 的 时 间 

internal static bool success = false; // 是 否 成 功 获得 天 气 信息 

private static WeatherInfo weatherResult=new WeatherInfo(); 
// 天 气 信息 


/* 根据 城市 名 称 获得 天 气 情况 。 

* 调用 方法 如 下 : 输入 参数 theCityName 城市 中 文 名 称 ， 如 深圳 ， 北 京 ; 

* theDayFlag 指定 是 当天 (1) ,明天 (2) 或 后 天 (3) ， 可 查询 未 来 三 天 的 天 气 情况 ; 

* 返回 数据 String[7] 一 个 一 维 数值 ， 共 有 7 个 元 素 ， 

* 从 [1] 到 [6] 分 别 表 示 城 市 ,天气 温度， 风向， 日期， 天 气 图 标 地 址 。 

*/ 

internal static void getWeather () 

{ 
// 每 隔 10 分 钟 刷新 一 次 天 气 信息 ， 如 果 当 前 时 间距 上 次 刷新 不 到 10 分 钟 ， 则 直接 返回 
if (success && lastGetTime .AddqMinutes (10) < DateTime .Now) 


return ; 
Console.WriteLine("get weather ..."); 
lastGetTime = DateTime.Now; 
try 


{ 
Var service = new weather.Service(); 
// 创 建 Web Service 客户 端 代理 
string [] result=service.getWeatherbyCityName 
(city, weather.theDayFlagEnum.Today); 
// 调 用 Web Service 得 到 天 气 信息 
30CCes5 = Erues 
buildResult (result); 


"i 
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catch 


weatherResult.clear (); 


weatherResult .weather = "未 能 获取 天 气 信息 "; 


success = false; 
} 
| 


// 将 Web Service 返回 的 string 数组 绑 定 到 WeatherInfo 对 象 
private static void buildResult (string[] result) 


下 


weatherResult .weather = result[2]; 
weatherResult.temprature = result [3]; 


weatherResult .wind 
weatherResult.date 


result[4]; 
result[5]; 


weatherResult.imgUrl = result[6]; 


public static WeatherInfo Weather 
{ 


b 


get { return weatherResult; } 


1 
/// <summary> 


/// 此 类 用 于 描述 天 气 信息 

/// </summary> 

internal class WeatherInfo 

{ 
public string weather = ""; 
public string temprature = ""; 
public string wind 3 
public string date = ""; 
public string imgUrl = ""; 
internal void clear() 


{ 


weather = 
temprature 


D 


// 天 气 (上 晴 、 阴 、 
// 温 度 
// 风 向 
// 日 期 
// 天 气 图 片 url 
// 清 除 天 气 信息 


多 云 等 ) 


(3) 在 表现 层 项 目 Nrem.Web 中 新 建 一 个 UserControls 文件 夹 ， 在 此 文件 夹 中 添加 一 


个 新 的 用 户 控件 Weather.ascx。 


(4) 在 Weatherascx 用 户 控件 中 用 一 个 table 控件 ， 显 示 城 市 名 称 和 天 气 信息 (文字 描 


述 和 图 片 )。 代 码 如 下 : 


<table> 
<EF> 


<td><asp:Label ID="cityName" runat="server" Text=" 城 市 名 称 "></asp:Label><br 


/> 天 气 </td> 

<td><asp:Image ID="weatherImage" 
AlternateText=" 天 气 " /></td> 

<td> 


runat="server™" 


<asp:Label ID="weatherDesc" runat="server" Text=" 天 气 情况 "></asp: 


Label></td> 
二 /让 开光 
</table> 


Height="32px" 


i 
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(5) 在 Weatherascx 用 户 控件 的 Page Load 事件 中 调用 WeatherService 类 ， 得 到 天 气 


信息 并 显示 于 页 面 上 。 


protected void Page Load (object sender, EventArgs e) 
{ 
if(!IsPostBack) 
cityName.Text = WeatherService.city; 
showWeather (); 
h 
private void showWeather () 
上. 
WeatherService.getWeather (); 
weatherDesc.Text = WeatherService.Weather.weather + "<br/>" 
+ WeatherService.Weather .temprature; // 显 示 天 气 文字 描述 
weatherImage.ImageUrl = WeatherService.Weather.imgUrl; // 显 示 天 气 图 片 


14.2.2 页 头 用 户 控件 


新 农 合 系统 的 页 面 头 部 用 于 显示 系统 名 称 、 当 前 时 间 、 当 前 用 户 、 天 气 信息 、 工 具 栏 ， 


其 中 天 气 信息 使 用 14.2.1 节 中 所 创建 的 Weatherascx 显示 。 新 农 合 页 面 头 部 也 被 封装 为 一 


个 


二 


户 控 件 Header.ascx， 控 件 外 观 如 图 14.6 所 示 。 


新 型 农村 合作 医疗 管理 信息 系统 


二 
15 让 


图 14.6 ”Header.ascx 用 户 控 件 外 观 


(1) 在 表现 层 项 目 Nrem.Web 中 添加 一 个 用 户 控件 Header.ascx， 页 面 代 码 如 下 : 


<%@ Control Language="C#" RutoEventWireup="true"” CodeBehind="Header.ascx. 
cs" Inherits="Nrcm.Web.UserControls.Header" $%> 

<%-- 注 册 天 气 预 报 用 户 控件 --%> 

<%@ Register src="Weather.ascx" tagname="Weather" tagprefix="ucl" $%> 
<div style="background:#f2f2fe; width:1000px; text-align:left;"> 
<%-- 工 具 栏 上 面 的 div--%> 

<div style="float:left"> 


<img src="../images/title.png" alt="" /> 
<span> 

当前 日 期 <%=DateTime.Now.ToLongDateString()%> 
当前 用 户 : 


<$%= Nrcem.Web.HttpCode.WebUtility.currentUser == null ? 
"未 知 用 户 ":Nrcm.Web .HttpCode .WebUtility.currentUser.UserName.Trim()%> 

【<a href=". ./LogoutPage .aspx"> 退 出 登录 </a>】 
</span> 
</div> 

<ucl:Weather ID="Weatherl" runat="server" /> <%-- 天 气 预 报 --%> 

</div> 
<div id="NavMenu"> <%-- 工 具 栏 --%> 
<div class="ImageMenu"><img src="../images/user.png" alt=" 农 民 档 案 "/><br/> 
<a href="../People/FamilyPage.aspx"> 农 民 参 合 档案 </a></div> 
<div class="ImageMenu"><img src="../images/medicine.png"” alt=" 药 品目 录 " 
/><br/> 
<a href="../Medicine/MedicinePage.aspx"> 药 品 检查 目录 </a></div> 
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<div class="ImageMenu"><img src="../images/hospital.png” alt=" 医 疗 机 构 " 
><PEX> 

<a href="../Hospital/HospitalPage.aspx"> 定 点 医疗 机 构 </a></div> 

<div class="ImageMenu"><img src="../images/policy.png” alt=" 报销 政策 " 
/><br/> 

<a href="../Policy/CopensateRatePage.aspx"> 报 销 政 策 管理 </a></div> 

<div class="ImageMenu"><img src="../images/dictionary.png”alt=" 字 典 维护 " 
/><br/> 

<a href="../Dictionary/DictionaryPage.aspx"> 数 据 字 典 维 护 </a></div> 

<div class="ImageMenu"><img src="../images/lock.png" alt=" 用 户 权限 " /><br/> 
<a href="../UserRight/ChangePassPage.aspx"> 用 户 权限 管理 </a></div> 

</div> 


(2) Header 用 户 控 件 工具 栏 中 “退出 登录 ”按钮 链接 到 LogoutPage.aspx 页 面 ， 此 页 
面 将 清空 Session 中 保存 的 登录 数据 。 页 面 代码 如 下 : 
<form id="forml" runat="server"> 
<div id="main"> 
<h3> 你 已 经 退出 新 农 合 管理 系统 ， 再 见 。</h3> 
点 击 <a href="Login.aspx"> 这 里 </a> 重 新 登录 <br /> 


</div> 
</form> 
protected void Page Load(object sender, EventArgs e) 
{ 
Nrcm.Web.HttpCode.WebUtility.currentUser = null; 
} 
14.2.3 ”和 母 版 页 


正如 本 节 开 始 部 分 所 述 ， 新 农 合 系统 采用 两 级 母 版 页 ， 其 中 主 母 版 页 内 容 较 为 简单 ， 
仅 包 含 页 头 控件 、 页 脚 控件 和 页 面 中 间 的 一 个 内 容 占 位 符 〈ContenPlaceHolder) 控件 。 主 
母 版 页 Nrcm.Master 页 面 代码 如 下 : 


<div id="main"> 
<form id="forml" runat="server"> 
<div> 
<ucl:Header ID="Headerl" runat="server" /> <g%-- 页 头 控件 --g> 
</div> 
<div> 
<asp:ContentPlaceHolder ID="content" runat="server"> 
<s%-- 内 容 占 位 符 --g%> 
</asp:ContentPlaceHolder> 
</div> 
<div> 
<uc2:Footer ID="Footerl" runat="server" /> <%-- 页 脚 控 件 --%> 
</div> 
</form> 
</div> 


14.3 基础 数据 管理 


新 农 合 系统 用 到 一 些 基础 数据 ， 包 括 在 很 多 管理 信息 系统 中 都 会 用 到 的 数据 字典 如 职 


SI 
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业 编 码 、 民 族人 代码、 行政 区 划 等 ， 还 包括 新 农 合 系统 所 特有 的 一 些 数据 ， 如 分 段 报销 比例 、 
报销 参数 等 。 新 农 合 管理 系统 需要 设计 一 些 页 面 以 允许 用 户 修改 这 些 数据 。 


14.3.1 数据 字典 管理 


管理 信息 系统 的 数据 库 中 一 些 属性 的 描述 通常 使 用 编码 而 不 是 文字 ， 例 如 ， 职 业 、 民 
族 、 健 康 状况 、 学 历 等 ， 新 农 合 系统 也 是 如 此 。 这 些 数据 字典 通常 允许 用 户 添 加 、 删 除 和 
修改 。 此 处 以 职业 为 例 说 明 如 何 设计 实现 简单 数据 管理 页 面 。 职 业 数据 保存 在 数据 库 的 
Occupation 表 中 ， 有 两 个 字段 职业 编号 ID 和 职业 名 称 Name， 两 字段 都 是 字符 类 型 。 

职业 管理 页 面 JobPage.aspx 布局 分 为 上 下 两 部 分 , 上 面 一 部 分 用 一 个 GridView 显示 目 
前 已 经 存在 的 职业 列表 ， 下 面 用 一 个 DetailsView 添加 新 职业 ，DetailsView 的 默认 显示 模 
式 为 插入 模式 , 页 面 上 使 用 SqlDataSource 作为 数据 源 .JobPage.aspx 页 面 设计 视图 如 图 14.7 
所 示 。 


职业 编 四 


Edit Delete Databound Databound 
Edit Delete Databound Databound 
Edit Delete Databound Databound 
Edit Delete Databound Databound 
Edit Delete Databound Databound 
Edit Delete Databound Databound 
Edit Delete Databound Databound 
Edit Delete Databound Databound 
Edit Delete Databound Databound 
Edit Delete Databound Dat abound 


座 加 

职业 编码 

职业 名 称 
Insert Cancel 
SallataSemree - SqlDatssourcel 


J 
则 


Bhi |O Split | Souree | [4][forsstorni) 


图 14.7 职业 管理 页 面 设计 视图 


JobPage.aspx 页 面 代码 如 下 : 


<form id="forml" runat="server"> 
<div> 
<s%-- 此 GridView 用 于 显示 职业 列表 --%> 
<asp:GridView ID="GridViewl" runat="server" AllowPaging="True" 
AutoGenerateColumns="False" DataKeyNames="ID" DataSourceID= 
"SqlDataSource1"> 
<Columns> 
<s--GridView 的 命令 列 ， 显 示 编辑 和 删除 命令 --$> 
<asp:CommandField ShowDeleteButton="True" ShowEditButton= 
"True" /> 
<asp:BoundField DataField="ID"” HeaderText=" 职 业 编 码 " ReadOnly= 
"True" 
SortExpression="ID"> 
<HeaderStyle Width="100px" /> 
</asp:BoundField> 
<asp:BoundField DataField="Name"” HeaderText=" 职 业 名 称 " 


Se 
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SortExpression="Name"> 
<HeaderStyle Width="200px" /> 
</asp:BoundField> 
</Columns> 
</asp:GridView> 
<s$-- 添 加 按钮 为 客户 端 按钮 ， 避 免 产 生 不 必要 的 页 面 回 发 --$> 
<input type="button" value=" 添 加 " id="addButton"/> 
<div id="insertView"> 
<$--DetailsView 用 于 添加 新 的 职业 ， 默 认 模 式 为 插入 模式 --%> 
<asp:DetailsView ID="DetailsViewl" runat="server" 
AutoGenerateRows="False" DataKeyNames="ID" DataSourceID= 
"SqlDataSourcel" 
DefaultMode="Insert"> 
<Fields> 
<asp:BoundField DataField="ID" HeaderText=" 职 业 编码 ”Readon1y= 
"True" 
SortExpression="ID" /> 
<asp:BoundField DataField="Name" HeaderText=" 职 业 名 称 " 
SortExpression="Name" /> 
<asp:CommandField ShowInsertButton="True" /> 
</Fields> 
</asp:DetailsView> 
</div> 
<$-- 鉴 于 职业 数据 非常 简单 ， 没 有 必要 三 层 结构 ， 直 接 使 用 SqlDataSource 实现 数据 访问 
==%> 
<asp:SqlDataSource ID="SqlDataSourcel" runat="server" 
Connectionstring="<%$ ConnectionStrings:NRCM %>" 
DeleteCommand="DELETE FROM [Occupation] WHERE [ID] = @ID" 
InsertCommand="INSERT INTO [Occupation] ([ID], [Name]) VALUES (@ID, 
@Name)" 
SelectCommand="SELECT [ID], [Name] FROM [Occupation]" 
UpdateCommand="UPDATE [Occupation] SET [Name] = @Name WHERE [ID] = 
QID"> 
<DeleteParameters> 
<asp:Parameter Name="ID" Type="String" /> 
</DeleteParameters> 
<UpdateParameters> 
<asp:Parameter Name="Name" Type="String" /> 
<asp:Parameter Name="ID" Type="String" /> 
</UpdateParameters> 
<InsertParameters> 
<asp:Parameter Nam 
<asp:Parameter Name: 
</InsertParameters> 
</asp:SqlDataSource> 
</div> 
</form> 


默认 情况 下 页 面 底部 的 添加 区 域 不 可 见 ， 只 有 当 用 户 单 击 “ 添 加 ”按钮 时 才 会 显示 。 
“添加 ”按钮 是 一 个 html 浏览 器 端 按钮 ， 不 会 引起 服务 器 端 回 发 ， 在 浏览 器 端 通过 JavaScr 
ipt 脚本 显示 添加 职业 的 区 域 。 


<script type="text/javascript" > 
stfunction (yy! 
$('#insertView') -hide (); // 隐 藏 添加 区 域 
$('#addButton') .click(function () { // 单 击 add 按钮 时 显示 添加 区 域 
$('#insertView') .show(); 
1); 
</script> 


"ID" Type="String" /> 
"Name" Type="String" /> 


-Sas 


14.3.2 行政 区 划 管理 


新 农 合 管理 系统 需要 对 行政 区 划 《〈 主 要 是 乡镇 和 村 ) 数据 进行 管理 。 各 级 行政 区 划 的 
维护 页 面 大致 相 同 ， 此 处 以 维护 乡镇 数据 为 例 说 明 页 面 如 何 实现 。 数 据 库 中 乡镇 表 Town 
包含 以 下 3 个 字段 。 

口 ID: nvarchar(10) 类 型 ， 乡 镇 编号 ， 主 键 。 

口 Name: nvarchar(20) 类 型 ， 乡 镇 名 称 。 

口 ParentID: 上 级 〈 即 所 属 区 县 ) ID， 外 键 关 联 到 区 县 表 County。 

(1) 在 实体 框架 项 目 Nrcm.Entity 中 添加 行政 区 划 数 据 的 实体 模型 DictionaryContext， 
数据 模型 内 容 如 图 14.8 所 示 。 


(Provinee 加 [于 car 本 ( 国 Comnty 5) = 因 Miare 


日 Properties 日 Preperties SProperties 日 Properties SProperties 
1 后 入 m 后 m 后 
了 re 可 rme 了 rme 本 rme 村 re 
CR 部 PwrentID 学 PurentID 部 PurentID 部 PurentID 
> Navigation Proper… Navigation Proper… 日 avigation Proper… Navigation Proper… 


图 14.8 行政 区 划 数 据 模型 


(2) 在 处 理 行 政 区 划 数 据 时 ， 经 常 需要 得 到 一 个 行政 区 划 的 整个 层次 结构 ， 例 如 ， 处 
理 村 级 数据 时 ， 需 要 知道 村 所 属 的 省 、 市 、 县 、 乡 。 为 了 描述 这 个 层次 结构 ， 在 业务 逻辑 
层 Nrem.Bll 项 目 中 添加 一 个 DistrictHierarchy 类 。 

// 行 政 区 划 层 次 数据 


public class DistrictHierarchy 


{ 


public string id=null; // 当 前 行政 区 划 ID 

public string name=null; // 当 前 行政 区 划 名 称 

public DistrictLevel level; // 当 前 行政 区 划 级 别 

public DistrictHierarchy parent=null; // 上 级 行政 区 划 层 次 
// 行 政 区 域 级 别 枚 举 类 型 


public enum DistrictLevel 


Province, City, County, Town, Village 


(3) 在 业务 逻辑 层 项 目 Nrcm.Bll 中 添加 一 个 DistrictUtility 类 ,此 类 主要 功能 为 获取 各 
级 行政 划 的 层次 结构 。 


public class DistrictUtility 

{ 
// 初 始 化 5 个 BLL 对 象 
CityBLL citybll = new CityBLL () ; 
ProvinceBLL provinceb11 = new ProvinceBLL () 
CountyBLL countybll = new CountyBLL () 
TownBLL townbll = new TownBLL () 
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VillageBLL villageb]l]l = new VillageBLL(); 
// 以 下 为 函数 字典 ， 通 过 行政 级 别 得 到 一 个 获得 该 行政 级 别 层次 结构 的 函数 
Dictionary<DistrictLevel, Func<string, DistrictHierarchy>> 
functionTable; 
public DistrictUtility() 
{ 
functionTable = new Dictionary<DistrictLevel, Func<string, 
DistrictHierarchy>>(); 
functionTable.Add (DistrictLevel .Province, getProvinceHierarchy); 
functionTable.Add (DistrictLevel .City, getCityHierarchy); 
functionTable.Add (DistrictLevel.County, getCountyHierarchy); 
functionTable.Add (DistrictLevel.Town, getTownHierarchy); 
functionTable.Add (DistrictLevel .Village, getVillageHierarchy); 
3 
// 根 据 行政 区 域 编号 和 级 别 获得 其 层次 结构 
public DistrictHierarchy getHierarchy (string districtID, DistrictLevel 
level) 
{ 
return functionTable[level] (districtID); // 调 用 函数 字典 中 的 相应 函数 
} 
// 得 到 省 级 行政 层次 结构 
private DistrictHierarchy getProvinceHierarchy (string id) 
{ 
var p = provincebll .getProvinceByID(id); 
if (p == null) 
throw new ApplicationException ("没有 找到 与 此 对 应 的 行政 区 域 。") ; 
return new DistrictHierarchy () 
{ id = p.ID, name = p.Name, parent = null, level=DistrictLevel. 
Province }; 
} 
// 得 到 市 级 行政 层次 结构 
private DistrictHierarchy getCityHierarchy (string id) 
{ 
City current = citybll .getCityById(id); 
if (current == null) 
throw new ApplicationException ("没有 找到 与 此 对 应 的 行政 区 域 。") ; 
DistrictHierarchy d = new DistrictHierarchy() 
{ id = current.ID, name = current.Name, level=DistrictLevel.City }; 
d.parent = getProvinceHierarchy (current .ParentID); 
// 得 到 其 上 一 级 层次 结构 
return d; 
} 
// 得 到 县 级 行政 层次 结构 
private DistrictHierarchy getCountyHierarchy (string id) 
i 
var current = countybll.getCountyById(id); 
if (current = null) 
throw new ApplicationException(" 没 有 找到 与 此 对 应 的 行政 区 域 。") ; 
DistrictHierarchy d = new DistrictHierarchy () 
{ id = current.ID, name = current.Name, level = DistrictLevel. 


County }; 
d.parent = getCityHierarchy (current.ParentID); 
return d; 

1 

// 得 到 乡镇 级 行政 层次 结构 


private DistrictHierarchy getTownHierarchy (string id) 


{ 
Var current = townbll .getTownById(id); 
if (current = null) 
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} 


throw new ApplicationException(" 没 有 找到 与 此 对 应 的 行政 区 域 。") ; 
DistrictHierarchy d = new DistrictHierarchy () 
{ id = current.ID, name = current.Name, level =DistrictLevel.Town }; 
d.parent = getCountyHierarchy (current.ParentID); 
return dz; 


} 
// 得 到 村 级 行政 层次 结构 


private DistrictHierarchy getVillageHierarchy (string id) 


i 


var current = villageb]l]l .getVillageById(id); 
if (current == null) 
throw new ApplicationException ("没有 找到 与 此 对 应 的 行政 区 域 。") ; 
DistrictHierarchy d = new DistrictHierarchy() 
{ id = current.ID, name = current.Name, level = DistrictLevel. 
Village }; 
d.parent = getTownHierarchy (current.ParentID); 
return d; 


(4) 在 数据 访问 层 项 目 Nrem.Dal 中 添加 一 个 类 TownDal， 实 现 乡 镇 数据 的 增删 改 查 


功能 。 


// 乡 镇 数据 访问 类 
public class TownDAL:ITownDAL 


{ 


arg) 


.524 。 
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<summary> 

得 到 指定 区 县 中 的 乡镇 列表 

</summary> 

<param name="countyId"> 区 县 ID</param> 
<param name="arg"> 分 页 参数 </param> 
<returns> 得 到 的 乡镇 列表 </returns> 


public List<Town> getTownsInCounty(string countyId, PageDataArgument 


{ 


using (DictionaryContext context = new DictionaryContext ()) 
{ 
// 编 写 LINQ 查询 代码 
Var query = (from c in context.Town 
where c.ParentID.Trim() == countyId 
orderby c.ID select c); 
if (arg.refreshCount) // 是 否 刷新 总 数 
arg.count = query.Count () 
// 根 据 分 页 参数 得 到 指定 页 的 数据 
var list = query.Skip(arg.pageIndex * arg.pageSize) .Take (arg. 
pageSize) .ToList(); 
EntityUtility.detachEntity(context, list); 
return list; 


1 


// 根 据 编号 得 到 乡镇 数据 
public Town getTownById(string id) 


{ 


using (DictionaryContext context = new DictionaryContext () ) 

{ 
return EntityUtility.selectOne<Town> (context .Town, t =>t.ID== 
id); 
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// 根 据 乡镇 名 称 得 到 某 区 县 中 的 乡镇 信息 〈 一 个 区 县 中 不 会 存在 同名 乡镇 ) 


public Town getTownByNameInCounty (string name, string CountyId) 


{ 


using (DictionaryContext context = new DictionaryContext()) 
{ 
return EntityUtility.selectOne<Town> (context .Town, 七 => 七 .Name 
== name && 七 .ParentID == countyId); 
世 
} 
// 添 加 乡镇 
public int addTown (Town town) 


{ 


using (DictionaryContext context = new DictionaryContext()) 
{ 
context .AddToTown (town); 
int n = context.SaveChanges (); 
EntityUtility.detachEntity(context, town); 
return n; 
} 
} 
// 根 据 乡 镇 编号 删除 乡镇 
public int deleteTown (string id) 
h 


using (DictionaryContext context = new DictionaryContext ()) 
Var town = (from 七 in context.Town 
where 七 .ID == id select t) .FirstOrDefault (); 
if (town == null) return 0; 
context .DeleteObject (town); 
return context.SaveChanges (); 
} 
} 
// 修 改 乡 镇 数据 
public int updateTown (Town town) 


using (DictionaryContext context = new DictionaryContext() ) 


{ 
return EntityUtility.update (context, town); 
} 


Th 


(5) 在 业务 罗 辑 层 Nrcm.Bll 中 添加 一 个 TownBl 类 ， 由 于 新 农 合 系统 中 没有 与 乡镇 相 
关 的 业务 罗 辑 ， 所 以 TownBll 类 只 是 调用 TownDal 类 的 相应 方法 ， 此 处 省 略 其 代码 。 
(6) 在 表现 层 项 目 Nrem.Web 中 添加 一 个 用 户 控件 SelectCounty.ascx, 此 控件 的 功能 是 
让 用 户 选择 一 个 区 县 ， 这 个 功能 在 整个 新 农 合 系统 中 用 到 多 次 ， 包 括 在 添加 和 修改 乡镇 时 
也 需要 选择 所 属 区 县 。 为 了 实现 复 用 ， 将 选择 区 县 的 功能 封装 成 一 个 用 户 控件 。 此 控件 中 
包含 3 个 级 联 下 拉 列 表 ， 分 别 表 示 省 、 市 和 区 县 ， 页 面 代码 如 下 : 


<asp:DropDownList ID="province" runat="server" AutoPostBack="true" 
DataValueField="ID" DataTextField="Name" 
onselectedindexchanged="province SelectedIndexChanged"> 

</asp:DropDownList> 

<asp:DropDownList ID="city" runat="server" AutoPostBack="true" 
DataValueField="ID" DataTextField="Name" 
onselectedindexchanged="city SelectedIndexChanged"> 

</asp:DropDownList> 

<asp:DropDownList ID="county" runat="server" DataValueField="ID" 
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DataTextField="Name"> 
</asp:DropDownList> 


(7) 在 用 户 控件 SelectCounty.ascx 的 Page Load 事件 中 ， 加 载 所 有 省 份 数据 。 
bool provinceLoaded = false; // 省 级 数据 是 否 已 经 加 载 


protected void Page Load (object sender, EventArgs e) 
{ 
if (!IsPostBack) 
initProvinces(); 
} 
private void initProvinces() 
{ 
if (provinceLoaded) return; // 如 果 数 据 已 经 被 加 载 则 返回 
provinceLoaded = true; 
// 得 到 所 有 省 份 数据 并 绑 定 到 下 拉 列 表 控 件 
var list = new ProvinceBLL() .getAllProvinces (PageDataArgument. 
allData); 
province.DataSource = list; 
province.DataBind(); 
// 在 下 拉 列 表 中 添加 一 个 "请 选择 "项 
province.Items.Insert (0，new ListItem("-- 请 选择 --"，"-1")); 
, 


(8) 在 用 户 控件 SelectCounty.ascx 省 级 下 拉 列 表 的 SelectedIndexChanged 事件 中 ， 加 
载 并 填充 所 选择 省 份 的 所 有 市 级 数据 。 


protected void province SelectedIndexChanged (object sender, EventArgs e) 


populateCity(); 


private void populateCity() 
{ 
string pid=province.SelectedValue; 
DropDownList control = city; 
if (string.IsNullOrEmpty (pid)) return; 
if (pid == "-1") return; 
List<City> list = new CityBLL() .getCitiesInProvince (pid, 
PageDataArgument .allData); 
control.DataSource = list; 
control .DataBind() 
ListItem item = new ListItem("-- 请 选择 --"，"-1"); 
control.Items.Insert (0, item); 


} 
(9) 在 用 户 控件 SelectCounty.ascx 城市 下 拉 列 表 的 SelectedIndexChanged 事件 中 ， 加 
载 并 填充 所 选择 城市 的 所 有 区 县 数据 。 


protected void city SelectedIndexChanged (object sender, EventArgs e) 
{ 


populateCounty (); 
} 
private void populateCounty () 
{ 
string pid = city.SelectedValue; 
if (string.IsNullOrEmpty (pid)) return; 
if (pid == "-1") return; 
var control = county; 
List<County> list = new CountyBLL() .getCountiesInCity(pid, new 
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(10) 在 


要 设置 省 、 趾 


jeDataArgument (0, 300, false)); 
trol.DataSource = list; 

trol.DataBind(); 

tItem item = new ListItem("-- 请 选择 --",，"-1"); 
trol.Items.Insert (0, item); 


用 户 控 件 SelectCounty.ascx 中 添加 一 个 属性 selectedCounty， 用 于 获取 和 设置 


当前 选中 的 区 县 ID。 当 读 取 该 属性 时 , 返回 区 县 下 拉 列 表 中 所 选中 的 值 。 当 设置 该 属性 时 ， 


和 、 县 各 个 下 拉 列 表 的 当前 选中 项 的 值 ， 必 要 时 还 需要 重新 加 载 数据 。 


// 获 取 或 设置 当前 选中 的 区 县 ID 


public 
1 
get 
{ 


bh 


Se 


string selectedCounty 


if (county.SelectedIndex > 0) 
return county.SelectedValue; 
else 
return null; 


// 当 设置 此 属性 (当前 选中 的 区 县 TD) 时， 需要 在 省 级 、 市 级 和 县 级 下 拉 列 表 中 都 显示 正确 
数据 


(11) 在 


// 如 果 省 份 未 加 载 则 加 载 ， 则 先 加 载 省 份 列表 
if (!(provinceLoaded || IsPostBack)) 
initProvinces(); 
string id = value; 
// 得 到 行政 区 划 层 次 
var hierarchy = new DistrictUtility() .getHierarchy (id, 
DistrictLevel .County); 
ListItem item=null; 
// 判 断 当前 所 选中 的 省 份 是 否 正 确 ， 否 则 改变 
if (hierarchy.parent.parent.id != province.SelectedValue) 
{ 
item = province.Items.FindByValue (hierarchy .parent .parent. 
id) ; 
if (item != null) item.Selected = true; 
PopulateCity() 


} 

// 判 断 当前 选中 的 市 级 行政 区 划 是 否 正确 ， 如 果 不 正 确 则 修改 

if (hierarchy.parent.id != city.SelectedValue) 

{ 
item = city.Items.FindByValue (hierarchy.parent .id) 
if (item != null) item.Selected = true; 
populateCounty (); 


} 

// 选 中 当前 县 

item = county.Items.FindByValue (id); 
if (item != null) item.Selected = true; 


表现 层 项 目 Nrem.Web 中 添加 一 个 页 面 TownPage.aspx， 用 于 显示 乡镇 列表 。 


页 面 布局 如 图 14.9 所 示 。 
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图 14.9 乡镇 列表 页 面 


TownPage.aspx 页 面 代 码 如 下 : 


<form id="forml" runat="server"> 
上 级 单位 : 
<ucl:SelectCounty ID="SelectCountyl" runat="server" /><%-- 选 择 区 县 用 户 控件 
= 
<asp:Button ID="filterButton" runat="server" Text=" 确 定 " onclick= 
"filterButton Click" />gnbsp;é&nbsp; 
<input id="addProvince" type="button" value=" 添 加 ” /> 
<%-- 乡 镇 列表 GridView--%> 
<asp:GridView ID="grid" runat="server" AutoGenerateColumns="false" 
DataKeyNames="ID" onrowcommand="grid RowCommand"> 
<Columns> 
<asp:BoundField DataField="ID" HeaderText=" 编 码 "> 
<HeaderStyle Width="200px" /> 
</asp:BoundField> 
<asp:BoundField DataField="Name" HeaderText=" 名 称 "> 
<HeaderStyle Width="200px" /> 
</asp:BoundField> 
<asp:TemplateField HeaderText=" 编 辑 "> 
<ItemTemplate> 
<img src=". ./images/edit.png" alt=" 编 辑 " onclick="edit (this)" 
style="cursor:pointer;" /> 
</ItemTemplate> 
</asp:TemplateField> 
<asp:TemplateField HeaderText=" 删 除 "> 
<ItemTemplate> 
<asp:ImageButton ID="ImageButtonl" ImageUrl="../images/ 
delete.png" 
OnClientClick="return $SjlUtility.confirmDelete()" 
CommandName="mydelete" 
runat="server" CommandArgument="'<%# Eval("ID") %$>' /> 
</ItemTemplate> 
</asp:TemplateField> 
<asp:TemplateField HeaderText=" 下 级 "> 
<ItemTemplate> 
<a href="VilligePage.aspx?parent=<%# Eval ("ID") %$>"> <%-- 查 看 下 级 
> 
<img src="../images/search.png"” alt=" 查 看 " style="border:none; 
height:16px;" /></a> 
</ItemTemplate> 
</asp:TemplateField> 
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</Columns> 
</asp:GridView> 
<webdiyer:AspNetPager ID="pager1" Tunat=" SerVer" 
onpagechanged="pagerl PageChanged"> 
</webdiyer:AspNetPager> 
</form> 


(12) 在 TownPage.aspx 页 面 的 Page_Load 事件 中 ， 根 据 QueryString 传递 过 来 的 区 县 
编号 加 载 乡 镇 列表 。 

protected void Page Load(object sender, EventArgs e) 

1 


if (!IsPostBack) 
{ 


} 


initData (); 


private void initData() 
{ 
pagerl.CurrentPageIndex = 1; 
string pid = Request.QueryString["parent"]; 
// 取 得 QueryString 中 的 区 县 编号 
// 如 果 QueryString 不 包含 区 县 编号 则 使 用 默认 区 县 
if (string.IsNullOrEmpty (pid)) 
pid = ConstValues.DefaultCounty; 
SelectCountyl.selectedCounty = pid; 
loadData (pid); 


| 
// 根 据 区 县 编号 加 载 乡镇 数据 
private void loadData (string parent) 
| 
if (string.IsNull1OrEmpty (Parent) || parent == "-1") 
return; 
TownBLL worker = new TownBLL(); 
PageDataArgument arg = new PageDataArgument (pagerl .CurrentPageIndex - 
1, pagerl.PageSize, true); 
grid.DataSource = worker.getTownsInCounty (parent, arg); 
pagerl .RecordCount = arg.count; 
grid.DataBind(); 


(13) 在 TownPage.aspx 页 面 的 “确定 ”按钮 中 , 根据 所 选择 的 区 县 重新 加 载 乡镇 数据 。 


private void loadData() 


{ 
loadData (SelectCounty]l .selectedCounty); 
’; 


(14) 在 TownPage.aspx 页 面 GridView 控件 的 RowCommand 事件 中 ， 如 果 用 户 选择 了 
删除 命令 ， 则 删除 当前 乡镇 。 


protected void grid RowCommand (object sender, GridViewCommandEventArgs e) 
i 
if (e.CommandName == "mydelete") 
TownBLL worker = new TownBLL(); 
worker.deleteTown (e.CommandArgument .ToString()); 
loadData(); 
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(15) 当 在 TownPage.aspx 页 面 单 击 “ 添 加 ”按钮 ， 或 者 在 GridView 中 选择 “编辑 ” 
命令 时 , 则 以 模式 窗口 打开 另外 一 个 页 面 TownEditPage.aspx, 并 在 其 中 编辑 乡镇 数据 。“ 添 
加 ”和 “编辑 ”在 浏览 端 通过 JavaScript 实现 ， 代 码 如 下 : 


<script type="text/javascript"> 
$ (function () { 
// 为 GridvView 添加 光 棒 效 果 
$('#grid') .find('tr') .hover( 
function () { $(this).addClass ("hoverRow"); }, 
function () { $(this) .removeClass ("hoverRow"); }); 
// 当 单 击 添加 按钮 时 执行 jnsert 函数 
$('#addProvince') .click (insert); 
1); //$ (function() {}) 
var dialogFeature = "dialogWidth:400px;dialogHeight:300px;"; 
// 以 模式 窗口 打开 TownEditPage .aspx 进行 数据 添加 
function insert() { 
if (window.showModalDialog ("TownEditPage.aspx",dialogFeature)) 
window.location.reload(); 


} 
// 编 辑 Gridview 中 所 选中 的 乡镇 
// 参 数 e 为 GridView 中 的 待 编辑 行 中 的 "编辑 "图 片 
function eqit(e) { 
var tr = $(e).closest('tr'); // 找 到 待 编 辑 行 
var id = tr.find('td') .eq(0) .html (); 
// 待 编辑 行 的 第 一 个 单元 格 内 容 即 为 乡镇 编号 
// 以 模式 窗口 打开 新 页 面 ， 然 后 重新 加 载 数 据 
if (window.showModalDialog ("TownEditPage.aspx?id=" + id, 
dialogFeature)) 
window.location.reload(); 


} 
</script> 


(16) 在 表现 层 项 目 Nrem.Web 中 添加 一 个 页 面 TownEditPage.aspx， 页 面 功能 为 编辑 
和 添加 乡镇 。 页 面 代 码 如 下 : 


<head runat="server"> 
<script type="text/javascript"> 
$ (function () { 
// 单 击 "取消 "按钮 时 关闭 对 话 框 ， 并 返回 False 
$('#cancel') .click (function(){ 
window.returnValue=false; window.close(); 
]) 7 
});//$ (function) 
</script> 
</head> 
<body> 
<form id="forml" runat="server"> 
<div id="editDialog" style="text-align:center;"> 
编码 : <asp:TextBox ID="cityCode" runat="server"></asp:TextBox> 
&nbsp; 
名 称 : <asp:TextBox ID="cityName" runat="server" ></asp:TextBox> 
BEE/ 
上 级 单位 : <ucl:SelectCounty ID="county" runat="server" /> 
<br /><br /> 
<asp:Button ID="OK" runat="server" Text=" 保 存 " onclick="OK Click" /> 
&nbsp; 


<input type="button" id="cancel" value=" 取 消 "/> 
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</div> 
</form> 
</body> 
public partial class TownEditPage : System.Web.UI.Page 
{ 
protected void Page Load(object sender, EventArgs e) 
‘ 
if (!IsPostBack) 
{ 
string oldid = Request.QueryString["id"] ?2 ""; 
// 得 到 所 编辑 的 乡镇 ID 
if (oldid.Length == 0) oldid = ConstValues.DefaultTown; 
if (oldid.Length > 0) 


a 
// 在 页 面 控件 上 显示 乡镇 信息 
var theTown = new TownBLL () .getTownById (oldid) 
county.selectedCounty = theTown.ParentID; 
cityCode.Text = theTown.ID; 
cityName.Text = theTown.Name; 


} 
} 
// 保 存 被 编辑 〈 或 添加 ) 的 乡镇 
private void save() 
{ 
string oldid=Request.QueryString["id"] ?2""; 
Town Vv = Town.CreateTown (cityCode.Text.Trim(), 
cityName.Text.Trim(), county.selectedCounty); 
TownBLL worker = new TownBLL(); 
if (oldid.Length > 0) 
{ 
// 主 键 未 改变 ， 则 修改 原 有 乡镇 数据 
if (oldid == cityCode .Text.Trim()) 
worker .updateTown (v) ; 
return; 
| 
worker.deleteTown (v.ID) ; // 主 键 改 变 ， 先 删除 再 添加 
} 


worker.addTown (v); // 添 加 乡镇 
日 


protected void OK _ Click(object sender, EventArgs e) 
是 
save(); 
// 通 过 JavaScript 关闭 对 话 框 并 返回 true 
this .ClientScript.RegisterStartupScript (this.GetType(), 
"closeDialog", 
“<script>window.returnValue=true;window.close();</script>"); 


| 


14.3.3 ”分 段 报销 比例 


新 农 合 采用 分 段 报销 的 方式 ， 参 合 农民 住院 费用 在 不 同 的 区 间 段 享受 不 同 的 报销 比 
例 , 如 500 元 以 下 不 报销 (报销 比例 为 0)，500 一 2000 元 报销 50%，2000 一 5000 报销 55%， 


人 
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依 此 类 推 。 分 段 报销 比例 中 的 分 界 点 和 各 段 报 销 比例 都 是 可 以 设置 的 。 分 段 报销 比例 保存 
在 数据 库 的 CompensateRangeRate 表 中 ， 此 表 共 有 以 下 4 个 字段 。 

口 了 PE: int 类 型 ， 主 键 ， 标 识 列 。 

口 Low: int 类 型 ， 分 段 起 点 。 

口 High: int 类 型 ， 分 段 终点 。 

口 Rate: float 类 型 ， 报 销 比 例 。 

(1) 在 实体 框架 层 Nrcm.Entity 项 目 中 , 添加 对 应 CompensateRangeRate 表 的 实体 模型 。 

(2) 在 数据 访问 层 Nrem.Dal 项 目 中 添加 一 个 类 CompensateRateDal， 编 写 相 关 数 据 访 


问 代码 。 


public static class CompensateRateDAL 


{ 


public static int add(Rate rate) 
{ 


return EntityUtility.add<MedicineContext, Rate>(rate); 


} 
public static int update (Rate rate) 


{ 


return EntityUtility.update<MedicineContext, Rate>(rate); 


! 


public static List<Rate> getAll() 


{ 


return EntityUtility.selectMany (new MedicineContext (). 
CompensateRangeRate, r => r.Low, PageDataArgument.allData); 


} 


public static int deleteByID(int id) 


{ 


return EntityUtility.delete (new MedicineContext () .Compensate 
RangeRate, r => r.ID == id); 


} 
; 


(3) 在 业务 逻辑 层 项 目 Nrcm.Bll 中 添加 一 个 类 CompensateRateBll， 这 个 类 的 功能 主 
要 是 调用 数据 访问 层 CompensateDal 类 的 相应 方法 ， 此 处 省 略 类 的 代码 。 
(4) 在 表现 层 项 目 Nrem.Web 中 添加 一 个 页 面 CompensateRatePage.aspx， 页 面 布局 如 


图 14.10 所 示 。 
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CompensateRatePage.aspx 页 面 上 部 使 用 GridView 显示 目前 已 有 的 分 段 报销 数据 ， 下 
部 使 用 DetailsView 进行 数据 添加 ， 使 用 ObjectDataSource 调用 业务 逻辑 层 CompensateRat 
eBl 类 实现 数据 增删 改 查 。 页 面 代码 如 下 : 


<asp:Content ID="Content2" ContentPlaceHolderID="content" runat="server"> 
<h3> 分 段 报 销 比 例 </h3> 
<$ 一 -分 段 报 销 比例 数据 列表 --%> 
<asp:GridView ID="gridl" runat="server" RutoGenerateColumns="False" 
CssClass="grid" DataKeyNames="ID"DataSourceID="ObjectDataSourcel" > 
<Columns> 
<asp:BoundField DataField="ID" HeaderText="ID" Readonly="True" 
Visible="false" InsertVisible="False" SortExpression="ID" /> 
<asp:BoundField DataField="Low" HeaderText=" 分 段 下 限 " ItemStyle- 
> 
</asp:BoundField> 
<asp:BoundField DataField="High"” HeaderText=" 分 段 上 限 " 
ItemStyle-Width="120" > 
</asp:BoundField> 
<asp:BoundField DataField="Rate" HeaderText=" 报 销 比例 " 
ItemStyle-Width="100"/> 
<asp:CommandField ButtonType="Image" ShowEditButton="true" 
EditImageUrl="../images/edit .png"” HeaderText=" 编 辑 ” /> 
<asp:CommandField ButtonType="Image" ShowDeleteButton="true" 
DeleteImageUrl="../images/delete.png" HeaderText=" 删 除 ” /> 
</Columns> 
</asp:GridView> 
<asp:ObjectDataSource ID="ObjectDataSourcel" runat="server" 
DataObjectTypeName="Nrcm.Entity.CompensateRangeRate" 
DeleteMethod="delete" 
InsertMethod="add" OldValuesParameterFormatSstring="original {0}" 
SelectMethod="getAll" TypeName="NTcm.B11.Medicine. 
CompensateRateBLL" 
UpdateMethod="update"> 
</asp:ObjectDataSource> 
<br /> <br /> 
<%--DetailsView 用 于 添加 新 的 分 段 数据 --%> 
<asp:DetailsView ID="DetailsViewl" runat="server" AutoGenerateRows= 
"False" 
DataKeyNames="ID" DataSourceID="ObjectDataSourcel" DefaultMode= 
"Inaert” “> 
<Fields> 
<asp:BoundField DataField="ID" HeaderText="ID" InsertVisible= 
"False" 
ReadOonly="True" SortExpression="ID" /> 
<asp:BoundField DataField="Low" HeaderText=" 分 段 下 限 " /> 
<asp:BoundField DataField="High" HeaderText=" 分 段 上 限 " /> 
<asp:BoundField DataField="Rate" HeaderText=" 报 销 比 例 " /> 
<asp:CommandField ButtonType="Button" ShowInsertButton="True" 
ShowCancelButton="false" /> 
</Fields> 
</asp:DetailsView> 
</asp:Content> 


(5) 新 农 合 采用 分 段 报销 制度 ， 即 对 于 一 个 住院 消费 金额 ， 并 不 是 乘 以 一 个 报销 比例 
就 得 到 报销 金额 ， 而 是 要 将 消费 总 额 划分 成 各 个 区 间 ， 每 个 区 间 享 受 不 同 的 报销 比例 ， 然 
后 将 各 个 区 间 的 报销 金额 相 加 ， 就 得 到 总 的 报销 金额 。 举 例 来 说 ， 分 段 报销 比例 为 500 元 


a 


第 3 篇 项 目 实战 


以 下 不 报销 报销 比例 为 0)，500-2000 元 报销 50%，2000-5000 报销 55%，5000-10000 
报销 60%，10000 以 上 报销 70%。 那 么 对 于 15000 元 的 住院 费用 报销 金额 为 
(500-0) X0+ (2000-500)X0.5+ (5000-2000) XO0.55+ (10000-5000) X0.6=5400 
在 新 农 合 系统 中 , 使 用 数据 库 中 的 自 定义 函数 实现 上 述 报销 金额 计算 功能 , 代码 如 下 : 


询 和 


CREATE FUNCTION [dbo] . [CalculateCompensation] 


=-- 根 据 分 段 报销 比例 计算 应 报销 金额 

(Qtotalmoney float) 一 -参数 :总 金额 
RETURNS float 一 返回 报销 金额 
AS 

BEGIN 


if @totalmoney is null return null; 

if @totalmoney=0 return 0; 

declare @remain float; 

declare @compensation float,@temp float; 

declare @low float,@rate float; 

set @low=-1; 

一 -确定 费用 所 属 的 最 高 报销 级 别 

select @low=low from CompensateRangeRate 

where low<=@totalmoney and high>=@totalMoney; 

if @low<0 return 0; 

Select @rate=rate from CompensateRangeRate where low=@low; 
set @compensation=(@totalmoney-@low+1)*@rate; 

-- 得 到 最 低级 别 至 次 高 级 别 应 得 的 报销 金额 

select etemp=sum( (high-low)*rate) from CompensateRangeRate 
where low<@low; 

set @compensation=@compensation+@temp 

return @compensation 

END 


14.4 ”家 庭 档案 管理 


新 农 合 参合 和 缴费 通常 以 家 庭 为 单位 ， 在 新 农 合 系统 中 需要 对 家 庭 和 农民 信息 进行 查 


维护 。 


14.4.1 数据 库 表 和 实体 类 


如 下 : 


与 农民 档案 相关 的 有 两 个 表 : 家 庭 表 FamilyInfo 和 人 员 表 PersonInfo。 两 个 表 结 构 
CREATE TABLE [dbo].[FamilyInfo] ( 
ID] [nvarchar] (20) NOT NULL, 一 -家 庭 编 号 
Name] [nvarchar] (10) NOT NULL, 一 -- 户 名 
Village] [nvarchar] (50) NOT NULL, =-- 所 属 村 庄 
Address] [nvarchar] (50) NULL, 一 -地 址 
Post] [nchar] (6) NULL, 一 -邮编 
Phone] [nvarchar] (20) NULL, 一 -联系 电话 
CMAttribute] [nchar] (2) NULL ， 一 -参合 属性 
FamilyAttribute] [nchar] (2) NULL ， 一 - 户 属性 


CONSTRAINT [PK FamilyInfo] PRIMARY KEY CLUSTERED 
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( ID] AsSC ) 

CREATE TABLE [dbol].[PersonInfo] ( 
ID] [nvarchar] (20) NOT NULL, 一 -个 人 编号 (身份 证 号 ) 
Name] [nvarchar] (10) NOT NULL, 一 -姓名 
Sex] [nchar] (1) NOT NULL, 一 性 别 
Birth] [datetime] NULL, 一 出 生日 期 
Nationality] [nchar] (2) NULL, 一 -民族 
[Marriage] [nchar] (2) NULL, 一 -婚姻 状况 
Health] [nchar] (2) NULL, 一 健康 状况 
[MedicalCard] [nvarchar] (20) NULL, 一 -参合 卡号 
Occupation] [nchar] (2) NULL, 一 -职业 
WorkAddress] [nvarchar] (50) NULL, -工作 单位 
Phone] [nvarchar] (20) NULL, -=-- 联 系 电话 
FamilyID] [nvarchar] (20) NOT NULL, -- 所 属 家 庭 编号 
FamilyRelation] [nchar] (2) NULL, 一 -与 户主 关系 
Village] [nvarchar] (10) NULL, 一 -所 属 村 庄 
CMAttribute] [nchar] (2) NULL, 一 -参合 属性 

CONSTRAINT [PK PersonInfo] PRIMARY KEY CLUSTERED 

0 ID] RSC ) 


在 FamilyInfo 表 中 有 一 个 Name 字段 ， 现 实生 活 中 并 没有 家 庭 名 称 这 个 说 法 ， 但 是 在 
新 农 合 系统 中 ， 当 显示 家 庭 列 表 时 ， 为 了 直观 地 表示 一 个 家 庭 ， 需 要 给 这 个 家 庭 一 个 唯一 
且 可 读 的 属性 ， 显 然 家 庭 编号 不 能 起 到 这 个 作用 。 因 为 家 庭 编号 全 是 字母 数字 ， 不 具有 可 
读 性 ， 所 以 给 FamilyInfo 表 中 添加 一 个 Name 字段 ， 以 直观 地 描述 家 庭 信 息 ， 这 个 字段 的 
值 通常 为 “ 张 三 家 ”、“ 李 四 户 ” 之 类 。 在 实体 框架 层 Nrem.Entity 项 目 中 添加 与 以 上 两 表 对 
应 的 数据 模型 ， 如 图 14.11 所 示 。 


®S FanilyInfo 


回 Properties 
铝 卫 
村 iane 
著 Yillage 
hddress 
加 Post Tationality 
车 Phone larriagee 
辐 cAttribute 村 Health 
车 FanilyAttribute 车 medicalCard 

GB Navigation Proper'… 车 occupation 
车 Yorkkddress 
村 Phone 
本 ranilyI 
FanilyRelation 
Village 
车 CAttribute 

困 Navigation Proper'… 


图 14.11 家 庭 和 个 人 实体 模型 


在 数据 库 中 的 FamilyInfo 和 PersonInfo 表 中 ， 职 业 、 民 族 、 健 康 状态 、 参 合 属性 等 字 
段 都 是 外 键 ， 其 中 保存 的 是 编码 ， 在 实体 类 中 这 些 属 性 的 值 也 是 编码 。 在 页 面 上 显示 这 些 
数据 时 ， 需 要 显示 为 可 读 的 文本 ， 而 不 是 编码 。 为 了 便于 将 数据 绑 定 到 GridView 等 数据 控 


a 
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件 ， 在 自动 生成 的 实体 类 中 ， 为 职业 、 民 族 等 每 一 个 编码 添加 一 个 属性 ， 表 示 此 编码 所 对 


应 的 文本 。 


修改 实体 模型 类 时 ， 要 注意 不 要 修改 自动 生成 的 文件 ， 因 为 这 些 自 动 生 成 的 文件 随时 
可 能 被 刷新 ， 在 其 中 的 修改 会 被 覆盖 。Visual Studio 自动 生成 的 实体 类 是 一 个 分 部 类 
(Partial Class)， 人 允许 把 一 个 类 的 代码 写 到 多 个 文件 中 。 如 果 要 对 自动 生成 的 实体 类 进行 修 


改 ， 应 该 添加 一 个 新 的 文件 ， 并 把 需要 添加 的 属性 或 方法 写 到 新 的 文件 中 。 
和 PersonInfo 类 的 扩展 代码 如 下 : 


public partial class FamilyInfo 
i 


对 FamilyInfo 


public string VillageName { get; set; } // 村 庄 名 称 
public string CMAttributeText { get; set; } // 参 合 属性 文本 
public string FamilyAttributeText { get; set; } // 家 庭 属性 文本 


public partial class PersonInfo 


public string villageName { get; set; } // 村 庄 名 称 
public string familyName { get; set; } // 户 名 
public string relationName { get; set; } // 与 户主 关系 
public string nationalityName { get; set; } // 民 族 名 称 
public string marriageState { get; set; } // 婚 姻 状况 
public string healthText { get; set; } // 健 康 状况 
public string occupationName { get; set; } // 职 业 名 称 
public string cmAttributeText { get; set; } // 参 合 属性 文本 


14.4.2 家庭 信息 管理 


在 新 农 合 系统 中 ， 可 以 浏览 某 一 村 的 所 有 家 庭 信息 ， 编 辑 、 删 除 这 些 家 庭 信息 ， 查 看 


家 庭 成 员 列 表 等 。 


(1) 在 数据 访问 层 项 目 Nrem.Dal 中 添加 一 个 FamilyDAL 类 ， 实 现 与 家 庭 信 息 相关 的 


数据 访问 功能 。 
public static class FamilyDAL 
* 


#region public methods 
// 添 加 家 庭 信息 
public static int addFamily (FamilyInfo family) 


} 


// 根 据 家 庭 ID 删除 家 庭 
public static int deleteByID (string id) 


| 
// 修 改 家 庭 信息 


public static int updateFamily (FamilyInfo family) 


return EntityUtility.add<PeopleContext, FamilyInfo> (family); 


return EntityUtility.delete( getQuery(), f => f.ID == id); 


return EntityUtility.update<PeopleContext, FamilyInfo> (family); 


“Se 
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// 根 据 ID 得 到 家 庭 信息 
public static FamilyInfo getByID(string id) 
下 
Var family= EntityUtility.selectOne (getContext () .FamilyInfo, f => 
fE.ID 一 id); 
loadDetail (family); 
return family; 
} 
// 得 到 某 村 庄 的 所 有 家 庭 
public static List<FamilyInfo> getFamiliesInVillage (string village, 
PageDataArgument pageArg) 


{ 
using (var context = getContext ()) 
{ 
var list=EntityUtility.selectMany (context .FamilyInfo, f => f. 
ID, pageArg, f => f.Village == village); 
list.ForEach(f => loadDetail (f)); 
return list; 
} 
#endregion 
#region private methods 
// 得 到 ObjectQuery 对 象 


private static ObjectQuery<FamilyInfo> getQuery() 
return new PeopleContext () .FamilyInfo; 


// 得 到 DataContext 对 象 
private static PeopleContext getContext () 


return new PeopleContext (); 


// 加 载 家 庭 详 细 信 息 〈 外 键 数据 ) 
private static void loadDetail (FamilyInfo family) 


if (family == null) 
return; 
// 得 到 家 庭 的 参合 属性 
if(family.CMAttribute!=null) 
family.CMAttributeText = SmallDictionaryDAL.getByID 
(SmallDictionaryTables.TableCMAttribute, family. 
CMAttribute) .name; 
// 得 到 家 庭 属 性 
if(family.FamilyAttribute!=null) 
family.FamilyAttributeText = SmallDictionaryDAL.getByID 
(SmallDictionaryTables.TableFamilyAttribute, family. 
FamilyAttribute) .name; 
// 得 到 村 庄 名 称 
if(family.Village!=null) 
family.VillageName = new VillageDAL() .getVillageById (family. 
Village) .Name; 
' 
#endregion 


} 

(2) 在 业务 逻辑 层 Nrcm.Bll 项 目 中 添加 一 个 FamilyBLL 类 。FamilyBLL 类 主要 功能 为 
FamilyDAL 类 的 相应 方法 ， 此 处 省 略 其 代码 。 

(3) 在 表现 层 项 目 Nrcm.Web 中 添加 一 个 页 面 FamilyPage.aspx， 页 面 布局 如 图 14.12 


和 


所 示 。 
Focploriomastor 
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图 14.12 家 庭 列表 页 面 布局 


FamilyPage.aspx 页 面 可 以 查看 某 一 村 庄 的 所 有 家 庭 列 表 。 页 面 最 项 部 为 一 个 用 户 控 
件 ， 此 用 户 控 件 包 含 一 组 级 联 下 拉 列 表 ， 可 以 选择 村 庄 。 用 户 控件 SelectVillage.ascx 代码 
如 下 : 


<table style="text-align:right;"> 

<tr> 

<td> 省 </td> 

<td><asp:DropDownList ID="province" runat="server" AutoPostBack="true" 
DataValueField="ID" DataTextField="Name" 
onselectedindexchanged="province SelectedIndexChanged"> 

</asp:DropDownList></td> 

<td> 市 </td> 

<td><asp:DropDownList ID="city" runat="server" AutoPostBack="true" 
DataValueField="ID" DataTextField="Name" 
onselectedindexchanged="city SelectedIndexChanged"> 

</asp:DropDownList></td> 

<td> 县 </td><td><asp:DropDownList ID="county" runat="server" 
DataValueField="ID" DataTextField="Name" AutoPostBack="true" 
onselectedindexchanged="county_SelectedIndexChanged" > 

</asp:DropDownList></td> 

</tr> 

<EES 

<td> 乡 (镇 ) </tq> 

<td><asp:DropDownList ID="town" runat="server" AutoPostBack="true" 
DataValueField="ID" DataTextField="Name" 
onselectedindexchanged="town SelectedIndexChanged"> 

</asp:DropDownList></td> 

<td> 村 ( 居 ) </td> 

<td><asp:DropDownList ID="village" runat="server" DataValueField="ID" 

DataTextField="Name" > 

</asp:DropDownList></td> 

<td></td><td></td> 

<HEE> 

</table> 
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在 图 14.12 所 示 的 FamilyPage.aspx 页 面 中 选中 村 庄 后 ， 单 击 “ 确 定 ” 按 钮 ， 即 可 在 页 
面 中 部 的 GridView 中 显示 此 村 所 有 户 。GridView 最 右 侧 三 列 为 命令 列 ， 单 击 相应 命令 可 
以 查看 此 户 成 员 列 表 、 编 辑 此 户 信息 、 删 除 此 户 。 页 面 底部 为 编辑 和 添加 户 的 区 域 ， 用 户 
输入 数据 ， 单 击 “ 保 存 ” 按 钮 ， 则 将 此 户 信 息 保存 到 数据 库 。FamilyPage.aspx 页 面 代码 
如 下 : 


<asp:Content ID="Content2" ContentPlaceHolderID="content" runat="server"> 
<ucl:SelectVillage ID="SelectVillagel" runat="server" /> 
<asp:Button ID="Button1" runat="server" onclick="Buttonl Click" Text=" 
查看 ”/> 
&nbsp; gnbsp; <asp:Button ID="addButton" runat="server"” Text=" 添 加 " 
onclick="addButton Click" /> 
<%-- 用 于 显示 家 庭 列 表 的 GridView--%> 
<asp:GridView ID="gridl" runat="server" AutoGenerateColumns="False" 
OnRowCommand="GridViewl RowCommand" DataKeyNames="ID" 
CssClass="grid"> 
<Columns> 
<asp:BoundField DataField="ID" HeaderText=" 家 庭 编号 ” /> 
<asp:BoundField DataField="Name" HeaderText=" 说 明 " /> 
<asp:BoundField DataField="VillageName" HeaderText=" 所 属 村 ( 居 ) 
"/> 
<asp:BoundField DataField="Address" HeaderText=" 地 址 " /> 
<asp:BoundField DataField="Post" HeaderText=" 邮 编 ” /> 
<asp:BoundField DataField="Phone" HeaderText=" 电 话 " /> 
<asp:BoundField DataField="CMAttributeText" HeaderText=" 参 合 属性 
nm /> 
<asp:BoundField DataField="FamilyAttribute" HeaderText=" 家 庭 属性 
De 
<asp:BoundField DataField="Village" Visible="false" /> 
<asp:BoundField DataField="CMAttribute" Visible="false" /> 
<asp:BoundField DataField="FamilyAttribute" Visible="false" /> 
<g%-- 以 下 列 为 模板 列 ， 包 含 一 个 图 片 链接 ， 链 接 到 另 一 个 页 面 显 示 家 庭 成 员 --%> 
<asp:TemplateField HeaderText=" 成 员 "> 
<ItemTemplate> 
<a href="../people/FamilyMember.aspx?family=<%#Eval ("ID")%>"> 
<img style="border:none;" src=". ./images/right2.png" alt=" 家 
庭 成 员 "/></a> 
</ItemTemplate> 
</asp:TemplateField> 
<&%-- 模 板 列 ， 包 含 一 个 图 片 按钮 ， 单 击 此 按钮 则 编辑 当前 家 庭 ， 通 过 JavaScript 
实现 --%> 
<asp:TemplateField HeaderText=" 编 辑 "> 
<ItemTemplate> 
<asp:ImageButton ID="editButton" runat="server" ImageUT1= 
"../images/edit.png" CommandName="edit0" CommandArgument= 
"<%#Eval ("ID")%®>' /> 
</ItemTemplate> 
</asp:TemplateField> 
<s-- 模 板 列 ， 包 含 一 个 图 片 按钮 ， 单 击 此 按钮 则 提示 确认 后 删除 当前 家 庭 --s> 
<asp:TemplateField HeaderText=" 删 除 "> 
<ItemTemplate> 
<asp:ImageButton ID="deleteButton" runat="server" ImageUT1= 
"../images/delete.png" CommandName="delete0" CommandArgument= 
'<%#Eval ("ID")%$>' OnClientClick="return deleteConfirm();" /> 
</ItemTemplate> 
</asp:TemplateField> 
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</Columns> 
</asp:GridView> 
<%--- 分 页 控件 --%> 
<webdiyer:AspNetPager 
ID="pager" runat="server" onpagechanged= 
"AspNetPagerl PageChanged"> 
</webdiyer:AspNetPager> 
<$-- 以 下 为 编辑 区 域 ， 可 以 编辑 和 添加 家 庭 信息 --g%> 
<asp:Panel ID="editPanel" runat="server" Visible="false"> 
<asp:Button ID="saveButton"” runat="server"” Text=" 保 存 " 
onclick="saveButton Click" />gnbsp; 
<asp:Button ID="cancelButton" runat="server" onclick= 
"cancelButton Click" 
Text=" 取 消 " /> <br /> 
<asp:HiddenField ID="hiddenID" runat="server" /> 
家 庭 编号 ，<asp:TextBox ID="familyID" runat="server"></asp:TextBox> 
家 庭 说 明 : <asp:TextBox ID="descriptoin" runat="server"></asp:TextBox> 
Sb /> 
详细 地 址 : <asp:TextBox ID="address" runat="server" width= 
"200px"></asp:TextBox><br /> 
联系 电话 : <asp:TextBox ID="phone" runat="server"></asp:TextBox> 
邮政 编码 : <asp:TextBox ID="post" runat="server"></asp:TextBox> <br /> 
村 ( 居 ) : <asp:DropDownList ID="villageList" runat="server" 
DataTextField="Name" 
DataValueField="id" Width="200"> 
</asp:DropDownList> 
<s%-- 单 击 以 下 图 片 按钮 则 以 模式 窗口 打开 另 一 页 面 ， 在 其 中 可 选择 村 庄 --%> 
<img ID="moreVillage" alt=" 更 多 村 ( 居 ) " src="../images/more.gif" 
style="cursor:pointer;" /> <br /> 
</asp:Panel> 


</asp:Content> 


(4) 在 FamilyPage.aspx 页 面 中 编写 JavaScript 代码 ， 当 单 击 “ 删 除 ” 按 钮 时 给 出 确认 
提示 ， 单 击 “ 选 择 村 庄 ” 图 片 按钮 时 打开 模式 对 话 框 选择 村 庄 。 


<script type="text/javascript"> 


$ (function() { 
$ ("#moreVillage") .click(selectVillage); 
]) 7 
var villageID = '#<%=villageList.ClientID %>'; // 村 庄 列表 控件 ID 


// 删 除 确认 


function deleteConfirm() { 


return confirm(" 确 实 要 删除 吗 ? ") ; 


// 打 开 模 式 对 话 框 ， 选 择 家 庭 所 属 村 庄 


function selectVillage() { 


.540 。 


var Village = $("#ct100 ct100 content content SelectVillagel 
village') [0] .value; 
if (!village) village = ""; 
Var a = window.showModalDialog("../Dialog/SelectVillagePage.aspx? 
village=" + village, 
null, "dialogWidth=500px;dialogHeight=200px;"); 
(a) // 如 果 对 话 框 没有 返回 结果 函数 返回 
return; 
// 对 话 框 的 返回 结果 形式 为 村 庄 ID， 村 庄 名 称 
// 将 返回 结果 从 有 逗号 〈，) 处 拆 开 ， 可 得 到 村 庄 ID 和 名 称 ， 将 其 显示 在 下 拉 列 表 中 
et | 1 le ey ee 
Nar DO’= “<option value=™ + si0l + > al) tt "</optlon>"s 
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) 


$ (villageID) .append(o); 
$ (villageID) .val (s[0]); 


</script> 
(5) 在 FamilyPage.aspx 页 面 上 “查看 ”按钮 的 Click 事件 中 ， 根 据 所 选择 的 村 庄 ， 读 
取 家 庭 数据 并 绑 定 到 GridView 控件 上 。 


protected void Button1 _ Click(object sender, EventArgs e) 


{ 


pager.CurrentPageIndex = 1; 
editPanel .Visible = false; 
bindGrid(); 
bindVillageList (); 


// 绑 定编 辑 区 域 的 村 庄 列表 
private void bindVillageList () 


有 


villageList.Items.Clear (); 
string id = SelectVillagel.selectedVillage; 
Village Vv = new VillageBLL() .getVillageById(id);  // 得 到 所 选择 的 村 庄 
if (v == null) return; 
PageDataArgument arg = PageDataArgument.allData; 
var list = new VillageBLL() .getVillagesInTown (v.ParentID, arg); 
// 得 到 同一 乡镇 的 所 有 村 
villageList.DataSource = list; 
villageList.DataBind(); 


} 
// 绑 定 家 庭 列表 
private void bindGrid() 


{ 


gridl.DataSource = null; 

gridl.DataBind(); 

string id = SelectVillagel.selectedVillage; 

if (string.IsNullOrEmpty(id)) return; 

PageDataArgument arg = new PageDataArgument (pager.CurrentPageIndex -1, 
pager.PageSize, true); 

pager.RecordCount = arg.count; 

var list = FamilyBLL.getFamiliesInVillage (id, arg); 

gridl.DataSource = list; 

gridl.DataBind(); 


(6) 在 FamilyPage.aspx 页 面 上 GridView 控件 的 RowCommand 事件 中 ,处 理 编辑 和 删 


除 命令 。 


//GridView 命令 处 理 程序 


protected void GridView1_RowCommand (object sender, GridViewCommand 
EventArgs e) 


// 本 页 面 GridView 支持 两 种 自 定义 命令 : 编辑 edit0 和 删除 delete0 
// 如 果 是 编辑 命令 ， 则 切换 到 编辑 模式 
if (e.CommandName == "edit0") 


人 
FamilyInfo family = FamilyBLL.getByID(e.CommandArgument. 
ToString()); 
if (family == null) return; 


// 将 待 编辑 家 庭 信息 显示 在 编辑 区 域 的 控件 上 
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familyID.Text = family.ID; 
descriptoin.Text = family.Name; 
phone.Text = family.Phone; 
post.Text = family.Post; 
villageList.SelectedValue = family.Village; 
editPanel .Visible = true; 

} 

// 如 果 是 删除 命令 ， 则 删除 指定 家 庭 

else if (e.CommandName == "delete0") 

{ 
FamilyBLL.deleteByID (e.CommandArgument .ToString ()); 
bindGrid(); 


(7) 在 FamilyPage.aspx 页 面 上 “添加 ”按钮 的 Click 事件 中 ， 清 空 并 显示 编辑 区 域 。 


protected void addButton Click(object sender, EventArgs e) 
editPanel .Visible = true; 
clearEdit (); 

} 

// 清 除 编辑 区 域内 容 

private void clearEdit () 


hiddenID.Value = "-1"; 
villageList.SelectedIndex = 0; 
familyID.Text = "" 


descriptoin.Text = ""; 
address.Text = ""; 
phone.Text 


post.Text = ;7 


(8) 在 FamilyPage.aspx 页 面 上 “保存 ”按钮 的 Click 事件 中 ， 保 存 当前 编辑 的 家 庭 信 
息 。 要 注意 区 分 当前 编辑 的 家 庭 是 新 增 家 庭 还 是 原 有 家 庭 , 相应 调用 Insert 和 Update 方法 。 


// 保 存 正在 编辑 的 家 庭 信息 
protected void saveButton Click(object sender, EventArgs e) 
{ 
// 根 据 编 辑 区 域 各 个 控件 的 值 构 建 一 个 FamilyInfo 对 象 
FamilyInfo family = new FamilyInfo(); 
family.ID = familyID.Text; 
family.Name = descriptoin.Text; 
family.Phone = phone.Text; 
family.Post = post.Text; 
family.Village = villageList.SelectedValue; 
/7 如 果 正在 编辑 的 家 庭 TD 为 -1 则 说 明 为 新 增 家 庭 ， 否 则 为 编辑 已 有 家 庭 
if (hiddenID.Value == "-1") 
FamilyBLL.addFamily (family); 
else 
FamilyBLL.updateFamily (family); 
editPanel.Visible = false; 
clearEdit (); 
bindGrid(); 
1} 


(9) 运行 FamilyPage.aspx， 运 行 界面 如 图 14.13 所 示 。 
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图 14.13 家庭 管理 页 面 


14.4.3 ”参合 农民 缴费 


参加 新 型 农村 合作 医疗 的 农民 必须 按照 国家 规定 的 数额 每 年 缴纳 一 次 费用 。 新 农 合 管 
理 系统 中 提供 缴费 功能 和 缴费 查询 功能 。 参 合 农民 缴费 数据 保存 在 数据 库 的 PersonPayment 
表 中 ， 表 结构 如 下 : 


CREATE TABLE [dbo] . [PersonPayment] ( 


[RecordID] [bigint] IDENTITY(1,1) NOT NULL, 一 -缴费 流水 号 ， 主 键 
[PersonID] [nvarchar] (20) NOT NULL, 一 -个 人 编号 
[PayDate] [datetime] NOT NULL DEFAULT (getdate () ) ，-- 缴 费 日 期 
[Year] [int] NOT NULL, =-- 缴 费 年 度 
[PayMoney] [money] NOT NULL, =-- 缴 费 金额 


CONSTRAINT [PK PersonPayment] PRIMARY KEY CLUSTERED 
( [RecordID] ASC ) 


当 农 民 缴费 时 ， 需 要 向 PersonPayment 表 中 添加 一 条 数据 ， 其 中 缴费 年 度 和 缴费 金额 
都 保存 在 数据 库 的 CompensateSpecialPolicy 表 中 。 参 合 农 民 缴 费 功能 由 数据 库 的 存储 过 程 
实现 ， 代 码 如 下 : 


ALTER PROCEDURE [dbo] . [sp PersonPay] 

@personid nvarchar (20) 一 -个 人 编号 

AS 

BEGIN 

set nocount on; 

declare @amount money; 

declare @year int; 

=-- 得 到 缴费 金额 和 参合 年 度 

select amount=[Value] from CompensateSpecialPolicy where [ID]="'05"'; 
select year=[Value] from CompensateSpecialPolicy where [ID]="04'"7 
insert into PersonPayment (PersonID,Year,PayMoney) 

values (@personid, @year,@amount); 
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set amount=0 

delete from PersonAccount where PersonId=@personid and Year=@year; 
declare @balancel money,@balance2 money 

-缴费 后 ， 重 新 设置 年 度 可 报 余额 

select Q@balancel=[Value] from CompensateSpecialPolicy where [ID]='01'; 
select Q@balance2=[Value] from CompensateSpecialPolicy where [ID]="'03'; 
insert into PersonAccount (PersonID, Year,HospitalBalance,ClinicBalance) 
values (epersonid, @year,@balancel,@balance2); 

=-- 设 置 参 合 属性 为 连续 参合 

update PersonInfo set cmAttribute ='03' where id=@personid; 

END 


(1) 在 数据 访问 层 项 目 Nrem.Dal 中 添加 一 个 类 PersonPaymentDAL， 代 码 如 下 : 


public static class PersonPaymentDAL 
| 
// 参 合 农民 缴费 ， 调 用 存储 过 程 实现 


public static void pay(string personid) 


Database db = DatabaseFactory.CreateDatabase ("NRCM"); 
db.ExecuteNonQuery ("sp_personpay",personid); 


// 得 到 某 人 某 年 的 缴费 信息 


public static PersonPayment getByPersonAndYear (string person, int year) 


return EntityUtility.selectOne (getQuery(), p => p.PersonID == 
person && p.Year == year); 


// 得 到 某 人 所 有 缴费 情况 


public static List<PersonPayment> getByPerson (string person) 


return EntityUtility.selectMany (getQuery(), p => p.Year, 
Nrcm.Common .PageDataArgument .allData, p => p.PersonID == 
person); 


private static System.Data.Objects.ObjectQuery<PersonPayment> 
getQuery () 


return new ExpenseContext() .PersonPayment; 
1 


(2) 在 业务 逻辑 层 项 目 Nrcm.Bll 中 添加 一 个 PersonPaymentBLL 类 ， 其 主要 功能 为 调 
日 数据 访问 层 PersonPaymentDAL 的 相应 功能 ， 此 处 省 略 其 代码 。 

(3) 在 表现 层 项 目 Nrcm.Web 中 添加 一 个 缴费 页 面 PersonPayPage.aspx， 页 面 布局 如 图 
14.14 所 示 。 

在 图 14.14 所 示 的 参合 缴费 页 面 中 ， 输 入 身份 证 号 ， 单 击 “ 查 看 ”按钮 ， 可 以 查看 此 

人 的 详细 信息 ， 以 避免 由 于 输入 错误 编号 而 误 给 他 人 缴费 。 页 面 下 部 显示 个 人 信息 的 区 域 
为 一 个 用 户 控件 PersonControl， 受 篇 幅 限 制 ， 此 处 不 给 出 该 控件 的 实现 过 程 ， 其 详细 代码 
可 参见 本 书 配套 光盘 。 在 PersonPayPage.aspx 页 面 中 单 击 “ 缴 费 ” 按 钮 即 可 完成 缴费 。 
PersonPayPage.aspx 页 面 代码 如 下 : 


mn 
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[errorWessage] 


人 详细 信息 


MiddenField - hiddsnyillage 


身份 证 号 姓名 


性 别  @ 男 C 女 出 生日 其 
参合 状态 [Unbound 。” ”到 参合 卡号 

家 庭 编 号 与 户主 关系 [Unbound ”加 
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| 而 两 页 可 一 一 Ri) 
4 | 
局 Desig |OSplit | Souree | [| Casp:Content#Content2> 


图 14.14 参合 缴费 页 面 布局 


<asp:Content ID="Content2" ContentPlaceHolderID="content" runat="server"> 


身份 证 号 : <asp:TextBox runat="server" ID="personID" /> 
<asp:Button runat="server"” ID="ok" Text=" 查 看 ”onclick="ok Click" /> 
<asp:Panel ID="payPanel" Visible="false" runat="server"> 
缴费 金额 : <asp :TextBox ID="money" runat="server" ReadOnly= 
"true"></asp:TextBox> 
<asp:Button ID="payButton" runat="server"” Text=" 缴 费 " onclick= 
BOEECnT Click™ /> 
<asp:Button ID="cancel" runat="server"” Text=" 取 消 " onclick= 
Scanceol Glick™ > 
</asp:Panel> 
xD > 
<asp:Label ID="errorMessage" runat="server" Text="" CssClass= 
"error"></asp:Label> <br /> 
<h3> 个 人 详细 信息 </h3> <hr /> 


<ucl:PersonControl ID="PersonControll" runat="server"/> 


</asp:Content> 
(4) 在 PersonPayPage.aspx 页 面 的 “查看 ”按钮 的 Click 事件 中 ,根据 输入 的 个 人 编写 
显示 个 人 详情 。 


protected void ok Click(object sender, EventArgs e) 


{ 


// 根 据 输入 的 个 人 编号 得 到 人 员 信息 
PersonInfo person = PersonBLL.getByID (personID.Text); 
if (person == null) 
下 
errorMessage.Visible = true; 
payPanel .Visible = false; 
errorMessage.Text = "未 能 找到 此 人 信息 。"; 
return; 
// 将 人 员 信 息 显示 在 页 面 底 部 的 PersonControl 控件 中 
PersonControll .person = person; 
int n = CompensateSpecialPolicyBLL.getPersonalPayment (); 
// 得 到 年 度 缴费 金额 


money.Text = n.ToString(); 
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payPanel .Visible = true; 
1 


(5) 在 PersonPayPage.aspx 页 面 的 “缴费 ”按钮 的 Click 事件 中 完成 缴费 。 


protected void Button1l Click(object sender, EventArgs e) 


PersonPaymentBLL.pay (personID.Text); // 调 用 缴费 方法 


clearInput (); // 清 除 用 户 输入 

// 向 客户 端 注册 JavaScript 显示 提示 信息 

string script = "<script>alert (' 缴 费 成 功 ! ') ;</script>"; 

ClientScript.RegisterStartupScript (this.GetType(), "success", 
script); 


(6) 在 表现 层 项 目 Nrem.Web 中 添加 一 个 页 面 PayQueryPage.aspx 以 实现 缴费 查询 功能 ， 
页 面 布 局 如 图 14.15 所 示 。 


PeopleFile.master 
如 必 档 管理 肌 人 证 号 : EJ 
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图 14.15 缴费 查询 页 面 


(7) 在 PayQueryPage.aspx 页 面 的 “查看 ”按钮 的 Click 事件 中 ， 根 据 用 户 所 输入 的 个 
人 编号 查询 个 人 信息 和 缴费 历史 记录 并 显示 在 页 面 上 。 


protected void ok Click(object sender, EventArgs e) 
{ 
PersonInfo person = PersonBLL.getByID (personID.Text) 
// 根 据 ID 得 到 个 人 信息 
// 如 果 此 人 不 存在 ， 则 隐藏 明细 信息 ， 方 法 返回 
if (person == null) 
' 
detailPanel .Visible=false; 
return; 
} 
// 如 果 此 人 存在 ， 则 显示 详细 信息 ， 并 显示 缴费 记录 
detailPanel .Visible=true; 
personDetail.person = person; 
Var list = PersonPaymentBLL.getByPerson (personID.Text); 
gridl.DataSource=list; 
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gridl.DataBind(); 


14.5 住院 费用 结算 和 审核 


参合 农民 在 定点 医疗 机 构 出 院 结算 时 ， 新 农 合 系统 需要 计算 可 报销 的 金额 ， 并 将 其 从 
总 费用 中 扣除 ， 农 民 只 需要 缴纳 剩余 部 分 的 费用 。 定 点 医疗 机 构 每 隔 一 段 时 间 去 新 农 合 部 
门 进行 结算 ， 集 中 领取 农民 住院 费用 中 新 农 合 报销 部 分 的 资金 。 


14.5.1 数据 库 表 结 构 


住院 费用 结算 是 新 农 合 系统 中 最 为 复杂 的 一 个 功能 模块 ， 涉 及 多 个 数据 库 表 和 复杂 的 
计算 规则 。 与 住院 费用 结算 相关 的 表 主 要 有 以 下 几 个 。 


1. 费用 项 目 表 


该 表 保存 了 医院 的 收费 项 目 (包括 药品 、 化 验 检查 、 手 术 、 材 料 费 等 ) 详细 信息 。 表 
结构 如 下 : 
CREATE TABLE [dbo] . [FYXM] ( 


XM CODE] [nvarchar] (15) NOT NULL, -- 项 目 编码 
XM NAME] [nvarchar] (15) NOT NULL, 一 -项 目 名 称 
JSM] [nvarchar] (10) NULL, 一 -拼音 检索 码 
PL CODE] [nchar] (2) NOT NULL, 一 品类 代码 
DJ] [float] NOT NULL, -单价 
BXBL] [float] NOT NULL DEFAULT ((0)), =-- 报 销 比例 
BXJB] [nchar] (2) NOT NULL DEFAULT ((0)), =-- 报 销 级 别 
YPGN CODE] [nchar] (2) NULL, 一 -药品 功能 代码 
GG] [nvarchar] (10) NULL, -- 规 格 
JX] [nvarchar] (10) NULL, 一 -剂型 
DW] [nvarchar] (10) NULL, 一 -药品 单位 
BZDW] [nvarchar] (10) NULL, =-- 包 装 单位 
BZSL] [int] NULL DEFAULT ((0) )， -- 包 装 数量 
CONSTRAINT [pk yyfyxm] PRIMARY KEY NONCLUSTERED 
( XM CODE] ASC ) 


2. 住院 费用 表 
该 表 保 存 了 医院 病人 的 消费 明细 数据 。 表 结构 如 下 : 


CREATE TABLE [dbo] . [PatientExpense] ( 


[RecordID] [bigint] IDENTITY(1,1) NOT NULL, 一 -流水 号 

[YYBM] [nvarchar] (10) NOT NULL, 一 医院 编码 

[ZYH] [nvarchar] (10) NOT NULL, 一 住院 号 

[FYID] [nvarchar] (20) NULL, 一 -费用 ID (HIS 中 的 主键 ) 
[RQ] [datetime] NULL, 一 -费用 日 期 

[XM CODE] [nvarchar] (15) NOT NULL, 一 -HIS 药品 编码 

[PL CODE] [nchar] (3) NULL, 一 品类 编码 
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DJ] [money] NOT NULL 
SL] [int] NOT NULL DEFAULT ((0) ) ， 

ZJE] [money] NOT NULL DEFAULT ((0)), 
MZZYBZ] [nchar] (1) NOT NULL, 

TPBZ] [nchar] (1) NULL DEFAULT (0°'), 

CZY CODE] [nvarchar] (10) NULL, 

SHBZ] [nchar] (2) NOT NULL DEFAULT ('00'), 
BXJE] [money] NOT NULL DEFAULT ((0)), 
DYXM] [nvarchar] (15) NULL, 

XM NAME] [nvarchar] (20) NULL, 

TranID] [bigint] NULL, 

YPJB] [nchar] (2) NULL, 

YPXJ] [money] NULL, 

BXBL] [float] NULL, 

YPSH] [bit] NULL, 


DEFAULT ((0)), 


一 单价 

一 -数量 

一 金额 

一 -门诊 住院 标志 
一 退 费 标志 

一 -操作 员 代 码 
一 审核 标志 

一 报销 金额 

一 对 应 于 新 农 合 的 药品 编码 
一 药品 名 称 

一 结算 业务 编号 
一 -药品 级 别 

一 药品 限 价 

一 -报销 比例 

一 是否 审核 


CONSTRAINT [PK PatientExpense 33D4B598] PRIMARY KEY CLUSTERED 


RecordID] ASC ) 


3. 住院 病人 表 
该 表 保存 了 病人 住院 的 相关 信息 ， 表 结构 如 下 : 


CREATE TABLE [dbo].[HospitalPatient]( 


YYBM] [nvarchar] (10) NOT NULL, 
ZYH] [nvarchar] (10) NOT NULL, 
RYRQ] [datetime] NULL, 

CYRQ] [datetime] NULL, 

KS CODE] [nchar] (2) NULL, 
SFZH] [nvarchar] (18) NOT NULL, 
XM] [nvarchar] (10) NOT NULL, 


XB] [nchar] (1) NULL, 

CSRQ] [datetime] NULL, 

ZY] [nvarchar] (10) NULL, 

DZ] [nvarchar] (24) NULL, 

DH] [nvarchar] (20) NULL, 

CW CODE] [nvarchar] (10) NULL, 

MZZYBZ] [nchar] (1) NULL ， 

CONSTRAINT [PK Patient] PRIMARY KEY CLUSTERED 
( YYBM] ASC, [ZYH] ASC ) 


4. 报销 结算 业务 表 


一 门诊 住院 标志 


该 表 保 存 了 住院 病人 结算 汇总 数据 ， 每 当 有 一 个 新 农 合 病人 出 院 结算 时 ， 就 会 产生 一 


条 汇总 数据 保存 到 此 表 。 表 结构 如 下 : 
CREATE TABLE [dbo] . [CompensationTransaction]( 

RecordID] [bigint] IDENTITY(1,1) NOT NULL, 一 -流水 号 
TranDate] [datetime] NOT NULL DEFAULT (getdate()), 一 -业务 发 生日 期 
SFZH] [nvarchar] (20) NOT NULL, 一 -身份 证 号 
YYBM] [nvarchar] (10) NOT NULL, 一 -医院 编码 
ZYH] [nvarchar] (10) NOT NULL, 一 住院 号 
MZZYBZ] [char] (1) NOT NULL, 一 -门诊 住院 标志 
FYZE] [money] NOT NULL DEFAULT ((0)), 一 -费用 总 额 
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KBJE] [money] NULL, 

SBJE] [money] NULL, 

SHJE] [money] NULL, 

SHBZ] [bit] NOT NULL DEFAULT ((0)), 
SHYH] [nvarchar] (10) NULL, 

SHRQ] [datetime] NULL, 

JSBZ] [bit] NOT NULL DEFAULT ((0)), 
JSID] [int] NULL, 


一 -可 报 金 额 
一 - 实 报 金额 
一 -审核 金额 
一 -审核 标志 
-- 审 核 用 户 
一 -审核 日 期 
一 -结算 标志 
一- 结算 编号 


CONSTRAINT [PK CompensationTransaction] PRIMARY KEY CLUSTERED 


( RecordID] nec 


5. 医疗 机 构 表 
该 表 保 存 了 各 个 定点 医疗 机 构 的 信息 ， 表 结构 如 下 : 


CREATE TABLE [dbo] . [Hospital]( 

ID] [nvarchar] (10) NOT NULL, 
Name] [nvarchar] (30) NOT NULL, 
HospitalLevel] [nchar] (2) NOT NULL, 
HospitalType] [nchar] (2) NOT NULL, 
Address] [nvarchar] (50) NULL, 
Post] [nvarchar] (10) NULL, 

Phone] [nvarchar] (15) NULL, 

CEO] [nvarchar] (10) NULL, 

Fund] [int] NULL, 

PersonnelNum] [int] NULL, 

BedNum] [int] NULL, 


CONSTRAINT [PK Hospital] PRIMARY KEY CLUSTERED 


( 


ID] ASC 
) 


14.5.2 ”住院 费用 结算 


各 个 定点 医疗 机 构 一 般 都 有 医院 管理 系统 (Hospital Information System， 简 称 HIS)， 


病人 住院 、 收 费 、 出 院 等 操作 都 通过 医院 的 HIS 系统 进行 管理 。 在 医院 就 诊 的 病人 有 参合 
农民 ， 也 有 其 他 病人 。 如 果 病 人 不 是 参合 农民 ， 那 么 病人 从 入 院 到 出 院 的 所 有 数据 都 与 新 
农 合 系统 无 关 。 如 果 病 人 是 参合 农民 ， 那 么 病人 出 院 结 算 时 ， 需 要 将 病人 住院 费用 导入 新 
农 合 数据 库 来 保存 ， 并 计算 报销 金额 。 住 院 费 用 报销 金额 的 计算 比较 规则 复杂 ， 以 下 为 计 


算 步 又 。 
(1) 得 到 病人 的 一 条 住院 费用 数据 。 此 数据 包含 了 使 月 


目 (为 了 简单 起 见 ， 下 文中 将 药品 和 检查 项 目 统称 为 药品 )， 


什么 药品 或 者 做 了 什么 检查 项 
单价 、 数量 、 金 额 分 别 是 多 少 。 


(2) 判断 这 种 药品 是 否 属于 可 报销 药品 。 新 农 合 系统 有 一 个 基本 用 药 目 录 ， 只 有 在 目 
录 中 的 药品 才 是 可 报销 药品 。 如 果 当 前 药品 不 是 可 报 药品 ， 则 报销 金额 为 0， 结 束 计 算 


过 程 。 
(3) 判断 这 种 药品 报销 级 别 和 医院 级 别 是 否 匹 配 。 每 种 


可 报销 药品 都 有 一 个 报销 级 别 ， 


例如 ， 某 种 药品 报销 级 别 为 县 级 医院 ， 则 说 明 只 有 县 级 以 上 医院 才能 使 用 这 种 药品 ， 如 果 
在 乡镇 医院 或 者 村 卫生 室 使 用 此 种 药品 则 不 予 报销 。 如 果 一 个 药品 是 可 报 药品 ， 需 要 检测 


.549 。 
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这 个 药品 的 报销 级 别 和 病人 所 住 医院 的 级 别 。 如 果 药 品 报销 级 别 高 于 医院 级 别 ， 则 报销 金 
额 为 0， 结束 计算 过 程 。 

(4) 检查 药品 限 价 。 每 种 可 报 药品 都 有 一 个 最 高 限 价 。 如 果 病 人 住院 时 使 用 此 药品 的 
价格 高 于 限 价 ， 则 按照 限 价 计算 报销 金额 ， 否 则 按照 实际 价格 计算 报销 金额 。 

(5) 得 到 药品 的 报销 比例 。 每 种 药品 都 可 以 设置 一 个 报销 比例 ， 是 全 额 还 是 按照 一 定 
比例 报销 。 将 第 4 步 得 到 的 药品 价格 乘 以 药品 数量 得 到 了 药品 金额 ， 再 乘 以 报销 比例 得 到 
报销 金额 。 

(6) 将 第 5 步 得 到 的 报销 金额 累加 到 总 报销 金额 中 。 

(7) 得 到 病人 下 一 条 住院 费用 数据 ， 转 第 〈2) 步 。 如 果 已 经 处 理 完 所 有 住院 费用 ， 
则 转 第 (8) 步 。 

(8) 根据 分 段 报 销 比例 和 总 报销 金额 ， 得 到 实际 报销 金额 。 

(9) 检查 此 人 当年 度 可 报 余额 ， 如 果 可 报 余额 不 足 ， 则 实际 报销 金额 为 当年 度 可 报 余 
额 。 新 农 合 系统 规定 ， 每 人 每 年 有 个 报销 上 限 。 报 销 上 限 减 去 当年 度 已 经 报销 的 金额 即 为 
当年 度 可 报 余额 。 

以 上 计算 报销 金额 的 过 程 涉及 许多 数据 库 操作 ， 为 了 提高 性 能 ， 在 新 农 合 系统 中 用 存 
储 过 程 来 实现 ， 存 储 过 程 代码 如 下 : 


-- Description: 计算 报销 费用 
=-- 找到 对 应 药品 ， 根 据 医院 等 级 、 药 品 限 价 和 报销 比例 ， 计 算 药 品 可 报 金 额 


ALTER PROCEDURE [dbo] . [CalculateExpenseCompensation] 


@yybm nvarchar (10) ， =-- 医 院 编码 

Q@zyh nvarchar (10) ， =-- 住 院 号 

@mzzybz nchar (1) -门诊 住院 标志 : 门诊 住院 
AS 
BEGIN 


set nocount on; 

=-- 首 先 检查 相关 参数 : 医院 、 病 人 、 费 用 

execute CheckHospitalPatient Q@yybm,@zyh,@mzzybz 

if @@error>0 return 

declare @yyjb nchar(2); 

select @yyjb=HospitalLevel from Hospital where ID=@yybm; 
一 -此 游标 针对 病人 费用 逐 行 读 取 ， 并 修改 对 应 项 目 、 药 品级 别 、 药 品 限 价 、 报 销 比例 、 报 销 金额 
declare mycursor cursor local forward only 

for select recordid,xm code,dj,sl,zje from PatientExpense 
where YYBM=@yybm and ZYH=@zyh AND MZZYBZ=@mzzybz 

for update of BXJE,DYXM,XM NAME,YPJB,YPXJ,BXBL,YPSH; 


declare @recordid bigint,@yyxm nvarchar(15); =-- 记 录 编 号 、 医 院 编码 
declare @dyxm nvarchar (15) ,xmmc nvarchar (20) 一 -对 应 项 目 、 项 目 名 称 
declare @dj money,@sl int ,ezje money; -- 单 价 、 数 量 、 金 额 
declare eprice limit money; 一 - 限 价 

declare ebxbl float,ebxjb nchar (2); =-- 报 销 比例 ， 报 销 级 别 
declare @shbz nchar (1) 一 -审核 标志 

open mycursor; 一 -打开 游标 并 循环 读 取 数据 


fetch next from mycursor into @recordid,@yyxm,@dj,@sl,@zje; 
while eefetch status=0 


begin 
update PatientExpense set dyxm=null]l,bxje=0,ypjb=null,ypxj=0,bxbl=0 
where current of mycursor; 一 -为 当前 行 设置 初 值 
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end 
END 


14.5.3 


参合 
数据 库 中 
过 的 数据 
住院 业务 
关 数 据 访 
(1) 


一 -找到 对 应 的 药品 项目 ) 
set @dyxm="" 
set @shbz="'0" 
select @dyxm=isnull (dyxm,''),@shbz=audit from MedicineMapping 
where YYBM=@yybm and YYXM=@yyxm; 
一 -未 找到 对 应 项 目 则 继续 下 条 费用 
if @dyxm="'' goto nextRow; 
if not exists 
(select * from fyxm where xm code=@dyxm) 
goto nextRow; 
select eprice limit = isnull(dj,0), @bxbl = isnull (bxbl,0), @bxjb= 
isnull (bxjb,'01'), @xmmc = xm name 
from fyxm where xm code=@dyxm; // 得 到 药品 限 价 
declare @ttt bit 
if @shbz="1" 
Set @ttt=1 
else 
set @ttt=0 
UPDATE PatientExpense 
set DYXM = @dyxm, xm name = @xmmc, ypjb = @bxjb, ypxj = @price limit, 
bxbl = @bxbl,ypsh = @ttt 
where current of mycursor; 


if @bxjb>@yyjb goto nextRow; =-- 药 品级 别 高 于 当前 医院 级 别 
if @shbz<>'1' goto nextrow; =-- 未 审核 
if @sl=0 begin set edj=ezje; set Q@sl=1; end  -- 未 输入 数量 则 默认 为 1 


if @dj>@price limit set @dj=@price limit; 

update PatientExpense set BXJE=@dj*@sl*@bxbl where current of mycursor; 
nextRow: 

fetch next from mycursor into @recordid,@yyxm,@dj,@s]l,@zje; 


-- 取 下 条 费用 数据 


住院 业务 审核 


病人 在 定点 医疗 机 构 出 院 结算 后 ， 病 人 的 住院 信息 和 费用 数据 就 被 导入 到 新 农 合 
。 为 了 防止 医院 做 假 ， 新 农 合 管理 部 门 需要 对 这 些 数据 进行 审核 ， 然 后 以 审核 通 
为 准 拨付 资金 给 定点 医疗 机 构 。 审 核 不 通过 的 数据 将 不 能 获得 新 农 合 报销 资金 。 
审核 涉及 两 个 页 面 : 一 个 汇总 页 面 和 一 个 明细 页 面 ， 下 面 将 说 明 这 两 个 页 面 及 相 
问 层 、 业 务 逻 辑 层 的 代码 。 

在 数据 访问 层 项 目 Nrem.Dal 中 添加 一 个 类 CompensationTransactionDAL， 实 现 与 


住院 结算 业务 相关 的 数据 访问 操作 ， 代 码 如 下 : 


public static class CompensationTransactionDAL 


{ 


// 根 据 业务 编号 得 到 业务 详情 
public static CompensationTransaction getById(long id) 


{ 
return EntityUtility.selectOne (getQuery() ,t=>t.RecordID==id); 


| 

// 得 到 指定 医院 指定 日 期 范围 内 的 业务 列表 

public static List<CompensationTransaction> getByHospitalAndDate 
(string hospital, DateRange date,PageDataArgument page) 

{ 


ls 
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ObjectQueryArgument<CompensationTransaction, long> arg=new 
ObjectQueryArgument<CompensationTransaction, long> (getQuery (), 
t=>t .RecordID, page); 
return EntityUtility.selectMany<CompensationTransaction, long> 
(arg, 七 => 七 .YYBM== hospital && t.TranDate >= date.from && t.TranDate 
<= date.to); 

| 

// 添 加 一 个 业务 

public static int add(CompensationTransaction tran) 

| 
return EntityUtility.add (new ExpenseContext () ，tran) ; 

} 

private static ObjectQuery<CompensationTransaction> getQuery() 

{ 
return new ExpenseContext () .CompensationTransaction; 


} 


(2) 在 业务 逻辑 层 项 目 Nrcm.Bll 中 添加 一 个 类 CompensationTransactionBLL。 
(3) 在 表现 层 项 目 Nrem.Web 中 添加 一 个 页 面 TransactionAuditPage.aspx， 页 面 布局 如 
图 14.16 所 示 。 
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图 14.16 住院 业务 审核 页 面 布 局 


在 图 14.16 所 示 的 TransactionAuditPage.aspx 页 面 中 , 在 页 面 项 部 选择 一 个 定点 医疗 机 
构 ， 输 入 一 个 日 期 范围 ， 单 击 “ 查 看 ”按钮 ， 即 可 查看 符合 条 件 的 住院 结算 业务 。 单 击 数 
据 列 表 中 的 “费用 明细 ”图 片 按钮 ， 即 可 打开 另外 一 个 页 面 ， 查 看 此 次 业务 的 费用 明细 。 
页 面 代码 如 下 : 


<asp:Content ID="Contentl" ContentPlaceHolderID="head" runat="server"> 


"3 


<gs-- 导 入 JavaScript 文件 和 CSS 文件 --%> 
<link href="../css/ui-lightness/jquery-ui-1.7.2.css" rel="stylesheet" 
type="text/css" /> 
<script type="text/javascript" src="../js/jquery-1.3.2.js"></script> 
<script src="../js/jquery-ui-l1.7.2.js" type="text/ 
javascript"></script> 
<script src="../js/MyUtility.js" type="text/javascript"></script> 
<script type="text/javascript"> 

$(function () { 


// 为 Griqview 添加 光 棒 效 果 
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D); 


$('table.grid') .find('tr') .hover( 
function () { $ (this).addClass ("hoverRow"); }, 
function () { $(this) .removeClass ("hoverRow"); }); 
$SjlUtility.addButtonClass(); 
// 使 用 jQuery 日 历 扩 展 两 个 日 期 文本 框 
$SjlUtility.jQueryDatePickerChinese(); 
$('#<$%=datel .ClientID$>') .datepicker (); 
$('#<%=date2.ClientID%>') .datepicker (); 

//$ (function) 


</script> 


</asp:Content> 
<asp:Content ID="Content2" ContentPlaceHolderID="content" runat="server"> 


<h3> 住 院 业 务 审核 </h3> 
医院 : <asp:DropDownList runat="server" ID="hospitalList" DataTextField= 
"Name" DataValueField="ID"></asp:DropDownList> 
日 期 范围 : <asp:TextBox runat="server" ID="datel" /> 
至 <asp:TextBox runat="server" ID="date2" /> 
<asp:Button Text=" 查 看 " runat="server" ID="ok" onclick="ok Click" /> 
<%-- 住 院 结算 业务 列表 --$> 
<asp:GridView ID="grid" runat="server" AutoGenerateColumns="False" 
DataKeyNames="RecordID" CssClass="grid" > 

<Columns> 


<asp:BoundField HeaderText=" 流 水 号 " DataField="RecordID" 
Visible="false" /> 

<asp:BoundField HeaderText=" 结 算 日 期 " DataField="TranDate" 
DataFormatstring="{0:d}" /> 

<asp:BoundField HeaderTex 
<asp:BoundField HeaderTex 


"住院 号 "” DataField="ZYH" /> 
身份 证 号 " DataField="SFZH" /> 
<asp:BoundField HeaderTex 费用 总 额 " DataField="FYZE" /> 
<asp:BoundField HeaderText=" 可 报 金额 " DataField="KBJE" /> 
<asp:BoundField HeaderText=" 实 报 金 额 " DataField="SBJE" /> 
<asp:CheckBoxField HeaderText=" 审 核 标志 " DataField="SHBZ" 
Readonly="true" /> 

<asp:BoundField HeaderText=" 审 核 金额 " DataField="SHJE" /> 
<asp:CheckBoxField HeaderText=" 结 算 标志 " DataField="JSBZ" 
Readonly="true" /> 

<asp:BoundField HeaderText=" 结 算 编号 " DataField="JSID" /> 
<g%-- 以 下 列 为 模板 列 ， 包 含 一 个 图 片 链接 ， 链 接 到 另 一 个 页 面 显示 费用 明细 --s%> 
<asp:TemplateField HeaderText=" 费 用 明细 "> 

<ItemTemplate> 

<a href='ExpenseDetialPage.aspx?tran=<%#Eval ("RecordID") $%>"' 
target=" blank"> 

<img style="border:none; width:20px;" src="../images/search. 
png”alt=" 费 用 明细 "/></a> 

</ItemTemplate> 

</asp:TemplateField> 


</Columns> 
<RowStyle HorizontalAlign="Center" /> 
</asp:GridView> 
<webdiyer:AspNetPager ID="pagerl" runat="server" onpagechanged= 
"pagerl PageChanged"> 
</webdiyer:AspNetPager> 


</asp:Content> 


(4) 在 TransactionAuditPage.aspx 页 面 的 Page Load 事件 中 ， 绑 定 医院 列表 。 


protected void Page Load (object sender, EventArgs e) 


-人 
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if (!IsPostBack) 
binqHospitals () > 
: 
// 绑 定 医院 列表 
private void binqHospitals () 
{ 
var list = HospitalBLL.getAll]l (PageDataArgument .allData); 
hospitalList.DataSource = list; 
hospitalList.DataBind(); 


(5) 在 TransactionAuditPage.aspx 页 面 上 “查看 ”按钮 的 Click 事件 中 ， 根 据 用 户 输入 


的 查询 条 件 获 取 数据 并 显示 在 页 面 上 。 


protected void ok Click(object sender, EventArgs e) 
€ 


Pagerl.CurrentPageIndex = 1; 
bindList (true); 

; 

private void binqdList (bool refreshCount) 


{ 


string hospital = hospitalList.SelectedValue; // 得 到 所 选中 的 医院 


DateRange date = DateRange.between2Date (datel .Text, date2.Text); 
// 获 得 日 期 范围 
PageDataArgument page = new PageDataArgument (); 
page.pageSize = pagerl .PageSize; 
page.refreshCount = refreshCount; 
page.pageIndex = pager1.CurrentPageIndex-17 
// 查 询 数据 并 绑 定 到 GridView 控件 
var list = CompensationTransactionBLL.getByHospitalAndDate 
(hospital, date, page); 
grid.DataSource = list; 
grid.DataBind(); 
3 
protected void pagerl PageChanged (object sender, EventArgs e) 
{ 


bindList (false); 


(6) 运行 TransactionAuditPage.aspx 页 面 ， 运 行 界面 如 图 14.17 所 示 。 
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图 14.17 住院 业务 查询 审核 页 面 
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(7) 在 数据 访问 层 项 目 Nrem.Dal 中 添加 一 个 PatientExpenseDAL 类 ， 实 现 与 病人 费用 
相关 的 数据 访问 功能 。 


public static class PatientExpenseDAL 
| 


/// <summary> 
/// 得 到 某 次 结算 业务 所 包含 的 费用 数据 
/// </summary> 
/// <param name="tran"> 结 算 业务 编号 </param> 
/// <param name="page"> 分 页 参数 </param> 
public static List<PatientExpense> getByTransaction (long tran, PageData 
Argument page) 
{ 
Var query = getQuery(); 
ObjectQueryArgument<PatientExpense, string> arg 
= new ObjectQueryArgument<PatientExpense, string>(); 
arg.query = query; 
arg.page = page; 
arg.sort = e=>e.FYID; 
var list=EntityUtility.selectMany (arg,e => e.TranID == tran); 
list.ForEach(e => loadDetail (e)); 
return list; 
} 
/// <summary> 
/// 审核 某 个 明细 费用 
/// </summary> 
/// <param name="recordId"> 费 用 记录 编号 </param> 
/// <param name="state"> 审 核 状态 </param> 
public static void audit (long recordId,AuditStateEnum state) 
{ 


string val=null; 
if (state == AuditStateEnum.None) 


val = “O00: 

else if (state == AuditStateEnum.Approve) 
val = “OL 

else if (state == AuditStateEnum.Deny) 
val = "02"; 

else 


throw new ApplicationException ("指定 的 审核 状态 不 合法 "); 
Var query = getQuery(); 
var item = (from ex in query 


where ex.RecordID == recordId 
select ex) .FirstOrDefault (); 
if (item == null) 


throw new ApplicationException(" 未 找到 费用 数据 ") ; 
item.SHBZ = val; 
query.Context .SaveChanges (); 
} 
/// <summary> 
/// 得 到 一 次 结算 的 审核 汇总 数据 并 保存 到 数据 库 
/// </summary> 
/// <param name="tranld"></param> 
/// <param name="user"></param> 
public static void auditSummary (long tranId, string user) 
{ 
using (ExpenseContext context = new ExpenseContext()) 
{ 
Database db = DatabaseFactory.CreateDatabase ("NRCM"); 
db.ExecuteNonQuery ("ShenHe", tranlId, user); /1/ 调 用 存储 过 程 


-Ms 


ShenHe 实现 


} 
} 
// 加 载 费 用 数据 的 外 键 信息 


private static void loadDetail (PatientExpense expense) 

人 
var item=FYxmDAL .getBYId (expense.XM CODE); 
expense-xmmc = item == null 2 "™ ;: item.XM NAME;> // 药 品名 称 
var audit = SmallDictionaryDAL.getByID (SmallDictionaryTables. 
TableAuditState, expense.SHBZ); 
expense.audit = audit == null ? "" : audit.name;  // 审 核 状 态 

lL 

private static ObjectQuery<PatientExpense> getQuery () 
return new ExpenseContext () .PatientExpense; 


} 


上 述 代 码 的 auditSummary0) 方 法 用 到 的 存储 过 程 ShenHe 代码 如 下 : 


CREATE PROCEDURE [dbo].[ShenHe] 
一 -审核 一 笔 业务 并 产生 相应 数据 


@tran bigint, 一 -被 审核 的 业务 ID 
@user nvarchar (10) 一 -审核 人 

AS 

BEGIN 


END 


(8) 


set nocount on; 
declare @je money 
set @je=0; 
一 -得 到 所 有 已 经 通过 审核 的 费用 总 额 
select @je=sum(isnull (bxje,0)) from PatientExpense 
where tranid=@tran and SHBZ="'01'; 
=-- 根 据 分 段 报销 比例 计算 应 得 报销 金额 
if @je is not null 
select @je=dbo.CalculateCompensation (@je); 
一 -保存 数据 
update CompensationTransaction 
set SHYH=@user,SHRQ=getdate(),SHBZ=1,SHJE=@je 
where RecordID=@tran; 


在 表现 层 项 目 Nrem.Web 中 添加 一 个 费用 明细 页 面 ExpenseDetialPage.aspx， 此 页 


面 可 以 显示 某 次 住院 结算 所 包含 的 所 有 费用 明细 ， 页 面 布局 如 图 14.18 所 示 。 


“be 


住院 费用 明细 


大 加 大 小 :|10 改变 页 面 大 小 | 全 部 通过 审核 。| 。 保存 审 术 数 据 | 
日 期 药品 项 目 单价 限 价 数量 全 额 审核 状 报销 金额 。 报销 比例 审核 


Databound Databound Databound Databound Databound Databound Databound Databound Databound 通过 驱 回 
Databound Databound Databound Databound Databound Databound Databound Databound Databound 通过 联 回 
Databound Databound Databound Databound Databound Databound Databound Databound Databound 通过 驱 回 


Databound Databound Databound Databound Databound Databound Databound Databound Databound 通过 联 回 


Databound Databound Databound Databound Databound Databound Databound Databhound Databound 通过 驱 回 


12345618910...>>> 
副 


a | ls] 


图 14.18 费用 明细 页 面 布 局 


第 14 章 新 农 合 管理 系统 


在 图 14.18 所 示 的 ExpenseDetialPage.aspx 页 面 中 ， 根 据 QueryString 中 传递 的 业务 编 


号 查找 此 业务 所 
面 的 “通过 ”或 


包含 的 所 有 费用 数据 并 显示 在 GridView 中 。 用 户 可 以 单 击 某 条 费用 数据 后 
“ 驭 回 ” 按 钮 实现 对 单条 费用 的 审核 功能 ， 也 可 以 单 击 “ 全 部 通过 审核 ” 


按钮 全 部 审核 当前 页 面 显示 的 所 有 费用 数据 。 用 户 单 击 “ 保 存 审核 数据 ”按钮 时 ， 将 对 此 
次 审核 数据 进行 汇总 ， 并 保存 到 数据 库 。ExpenseDetialPage.aspx 页 面 代 码 如 下 : 


<form id="forml" runat="server"> 


<h3> 住 院 费 用 明细 </h3> 


<div> 


页 面 大 小 : <asp:TextBox runat="server" ID="pageSize" Text="10" /> 
<asp:Button runat="server"” ID="ok" Text=" 改 变 页 面 大 小 " onclick= 
sok CEiek" /> 
<asp:Button runat="server" ID="approveAll"” Text=" 全 部 通过 审核 " 
onclick="approveAll Click" /> 
<asp:Button runat="server"” ID="audit" Text=" 保 存 审 核 数 据 " onclick= 
"audit Click" /><br /> 
<asp:GridView ID="grid" runat="server" CssClass="grid" 
CellPadding="8" 
AutoGenerateColumns="False" DataKeyNames="RecordID" 
onrowcommand="grid RowCommand"> 
<Columns> 


<asp:BoundField DataField="RecordID" HeaderText="RecordID" 
ReadOonly="True" Visible="false"/> 

<asp:BoundField DataField="RQ" HeaderText=" 日 期 " 
DataFormatString="{0:g}j"/> 

<asp:BoundField DataField="XM NAME" HeaderText=" 药 品 项 目 " /> 
<asp:BoundField DataField="DJ" HeaderText=" 单 价 " /> 
<asp:BoundField DataField="YPXJ" HeaderText=" 限 价 "/> 
<asp:BoundField DataField="SL" HeaderText=" 数 量 " /> 
<asp:BoundField DataField="ZJE"” HeaderText=" 人 金额 " /> 
<asp:BoundField DataField="audit" HeaderText=" 审 核 状态 ”/> 
<asp:BoundField BXJE"” HeaderText=" 报 销 金 额 ” /> 
<asp:BoundField BXBL"” HeaderText=" 报 销 比例 "” /> 
<% 一 以 下 模板 列 中 包含 两 个 命令 按钮 一 个 "通过 "和 一 个 "了 驱 回 "--%> 
<asp:TemplateField HeaderText=" 审 核 "> 

<ItemTemplate> 

<asp:LinkButton ID="approve" runat="server" CommandName= 
"approve" CommandRrgument='<s#Eval ("RecordID")%>' Text=" 通 过 " 
/> 

<asp:LinkButton ID="deny" runat="server" CommandName="deny" 
Text=" 了 驳回" CommandArgument="'<%#Eval ("RecordID")%$>' /> 
</ItemTemplate> 

</asp:TemplateField> 


</Columns> 
</asp:GridView> 
<webdiyer:AspNetPager ID="pagerl" runat="server" 
onpagechanged="pager1l PageChanged"> 
</webdiyer:AspNetPager> 


</div> 
</form> 


(9) 在 ExpenseDetialPage.aspx 页 面 的 Page Load 事件 中 ， 查 询 费用 明细 数据 并 显示 。 


protected void Page Load (object sender, EventArgs e) 


{ 


if (!IsPostBack) 


i 


第 3 篇 项 目 实战 


件 


i 
bindList (true); 
} 
! 
/// <summary> 
/// 绑 定 费用 明细 数据 列表 
/// </summary> 
/// <param name="firstTime"> 是 否 第 一 次 加 载 数据 ， 如 果 是 则 需要 刷新 总 记录 数 </param> 
private void bindList(bool firstTime) 
{ 
string tran = Request.QueryString["tran"]; // 得 到 业务 编号 
long id; 
if (!long.TryParse (tran, out id)) 
return; 
// 根 据 分 页 控件 得 到 分 页 参数 
PageDataArgument page = new PageDataArgument () ; 
page.pageSize = pagerl .PageSize; 
page.refreshCount = true; 
page.pageIndex = pagerl.CurrentPageIndex - 1; 
// 执 行 查询 并 绑 定数 据 
List<PatientExpense> list = PatientExpenseBLL .getByTransaction (id, 
page); 
if (firstTime) 
pagerl .RecordCount=page.count; 
grid.DataSource = list; 
grid.DataBind(); 
时 


(10) 在 ExpenseDetialPage.aspx 页 面 上 “改变 页 面 大 小 ”按钮 的 Click 事件 和 分 页 控 
的 PageChanged 事件 中 ， 根 据 新 的 页 面 大 小 和 页 码 绑 定 数据 。 


protected void pagerl PageChanged (object sender, EventArgs e) 
{ 


} 
protected void ok Click(object sender, EventArgs e) 


1 


bindList (false); 


pagerl .PageSize = int.Parse (pageSize.Text); 
bindList (false); 
有 


1 


(11) 在 ExpenseDetialPage.aspx 页 面 上 GridView 控件 的 RowCommand 事件 中 ， 对 当 


费用 进行 审核 操作 (通过 或 驶 回 )。 


protected void grid RowCommand (object sender, GridViewCommandEventArgs e) 
{ 
AuditstateEnum audit = AuditStateEnum.Invalid ; 
// 根 据 CommandName 设置 审核 状态 
if (e.CommandName == "approve") 
audit = AuditSstateEnum.Approve; 
else if (e.CommandName == "deny") 
audit = AuditStateEnum.Deny; 
long id = Convert.ToInt64 (e.CommandArgument); 
PatientExpenseBLL.audit (id, audit); 
bindList (false); // 重 新 绑 定 数据 
} 
public enum RuditStateEnum 
{ None, Approve, Deny, Invalid } 
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(12) 在 ExpenseDetialPage.aspx 页 面 上 “全 部 通过 审核 ”按钮 的 Click 


面 上 显示 的 所 有 费用 都 设置 为 审核 通过 。 


protected void approveAll Click(object sender, EventArgs e) 


Nl 


有 件 中 ， 


{ 
for {int i = 0; i < grid.Rows.Count; i++) 
{ 
long id = Convert.ToInt64(grid.DataKeys[i][0]); 
PatientExpenseBLL.audit (id, AuditStateEnum.Approve); 
} 
bindList (false); 
} protected void approveAll Click(object sender, EventArgs e) 
‘ 
// 对 GridView 的 所 有 行进 行 循环 ， 设 置 每 行 数据 的 审核 状态 为 通过 
for (int i = 0; i < grid.Rows.Count; i++) 
{ 
long id = Convert.ToInt64 (grid.DataKeys[i] [0]); 
PatientExpenseBLL.audit (id, AuditSstateEnum.Approve); 
F 
bindList (false); // 重 新 绑 定 数据 
} 
(13) 在 ExpenseDetialPage.aspx 页 面 上 “保存 审核 数据 ”按钮 的 Click 事件 中 ， 
核 汇总 数据 保存 到 数据 库 。 
protected void audit Click(object sender, EventArgs e) 
|! 
string tran = Request.QueryString["tran"]; 
long id; 
if (!long.TryParse (tran, out id)) 
return; 
PatientExpenseBLL.auditSummary (id, WebUtility.currentUser.ID); 
(14) 在 浏览 器 中 查看 ExpenseDetialPage.aspx 页 面 ， 运 行 界面 如 图 14.19 所 示 。 


文件 中。 编辑 四 查看 VW) 历史 GG) 书签 中 ”工具 0 和 助 0 

cx tt 
come 可 汪 -f- 可 -MM- 回 介 - 避 - 自 -» 
http://localhose. aspz?tran=25| ~ - 
住院 费用 明细 加 
页 面 大 小 : [in 改变 页 面 大 小 |“ 全 部 通过 审核 】 “保存 窜 核 数据 

日 其 药品 项 目 恨 价 。 数量 审核 状态 ”报销 全 颜 报销 比例 。 审核 

2004-8-31 11:07 一 次 性 材料 0.0000 0.0000 5.0000 ”通过 0.0000 W 通过 联 回 

2004-8-31 11:07 小儿 头皮 针 和 输液 4.0000 4.0000 4.0000 ”通过 4.0000 1 通过 联 回 

2004-8-31 11:07 住院 诊疗 费 2.0000 2.0000 2.0000 ”通过 2.0000 1 通过 联 回 

2004 8 31 11:07 一 级 护理 3.0000 3.0000 3.0000 ”通过 3.0000 1 通过 联 回 

2004-8-31 11:07 普通 床位 费 10.0000 10.0000 10. 0000 通过 10.0000 1 通过 驶 回 

2004-8-31 11:07 治疗 费 10.0000 10.0000 10.0000 通过 5.0000 0.5 通过 颈 回 

2004-8-31 11:07 10%65 2.1000 2.1000 4.2000 ”通过 4.2000 4 通过 联 回 

2004-8-31 11:07 ”病毒 只 0.4200 0.4200 0.8400 ”通过 0.8400 » 通过 颈 回 

2004-8-31 11:07 维生素 C 0.4200 0.4200 2.5200 ”通过 2.5200 如 通过 联 回 

2004-8-31 11:07 10%6S 2.1000 2.1000 4.2000 ”通过 4.2000 各 通过 驱 回 
e123456189>)> 司 
Ea Ea 


图 14.19 费用 明细 审核 页 面 


将 页 
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14.6 小 结 


本 章 介绍 了 新 型 农村 合作 医疗 管理 信息 系统 的 设计 与 实现 。 新 农 合 系统 业务 较为 复 
杂 ， 功 能 模块 较 多 ， 受 篇 幅 限 制 ， 本 章 仅 介绍 了 几 个 典型 的 功能 模块 ， 此 项 目 完整 代码 可 
参照 配套 光盘 。 读 者 可 通过 光盘 中 的 源 代 码 ， 了 解 更 多 的 开发 技巧 。 
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附录 “Visual Studio 操作 快捷 键 


本 附录 主要 是 根据 一 些 开 发 人 员 的 工作 经 验 而 总 结 的 ， 目 的 就 是 增加 开发 速度 和 效 
率 。 下 面 是 一 些 常用 快捷 键 ， 列 表 如 下 所 示 。 


快 捷 键 功 能 
F12 转 到 定义 
Fl 帮助 
F5 运行 程序 并 调试 
F10 跨 过 程序 执行 
F7 切换 到 代码 或 者 ASPX 
F4 查看 属性 
Fl11 逐 语 句 调试 
F10 逐 过 程 调试 
Ctrl +KCtrlHD 格式 化 文档 
Ctrl +KCtrl+C 注释 代码 段 
Ctrl +KCtrlHU 取消 注释 
Ctrl +KCtrl+L 取消 标签 
Ctrl +MCtrlHM 折 释 代码 段 
Alt+F4 关闭 Visual Studio 开发 工具 界面 
Ctrl +F4 
Shift+F5 
CtrlHJ 
Altt 一 择 最 接近 的 名 称 
Ctrl +L 删除 选 定 行 
Ctrl + Shift + 空格 出 现 参数 列表 提示 
Ctrl 十 R 十 E 把 字段 封装 成 属性 
Ctrl 十 Shift 十 B 编译 程序 
Cttl 十 左右 方向 键 按 单词 移动 光标 ， 准 确 快速 
Ctrl +R 十 R 重 命名 变量 、 方 法 等 的 名 称 
CtrlHB 定义 书签 
CtrlHN 转 到 下 一 个 书签 (方便 修改 代码 ) 
Shift+F7 切换 代码 和 页 面 
AltHB+E 重新 生成 
CtrlHHEF7 生成 编译 
CtrlHO 打开 文件 
Ctrl+ShiftrO 打开 项 目 


CtrlHShiftHC 显示 类 视图 窗口 


附录 Visual Studio 操作 快捷 键 


( 续 表 ) 

快 捷 键 功 能 
Shift+F4 显 bE 
Ctrl+Shift+E 显示 资源 视图 
CtrlHF12 转 到 声明 
CtrlHAltHJ 对 象 浏览 
CtrlHAlttF1 帮助 目录 
Ctrl+F1 动态 帮助 
Shift+F1 当前 窗口 帮助 
CtrlitAlt+F3 帮助 一 一 搜索 
Shift+AIttENTER 全 屏 显示 
Ctrl+- 向 后 定位 
Ctrl+Shift+- 向 前 定位 
CtrlHF4 关闭 文档 窗口 
Ctrl+Page Down 光标 定位 到 窗口 上 方 
Ctrl+Page Up 光标 定位 到 窗口 下 方 
CtrlHTab 下 一 个 文档 窗口 


CtrltrK, Ctrl+tL 
CtritrK, Ctrl+C 
Ctrl+ 玉 ，Ctrl+HU 
Ctrl+HM，Ctrl+O 
CtrlHM，Ctrl+L 


取消 remark 

注释 选择 的 代码 

取消 对 选择 代码 的 注释 
折 芭 代码 定义 


CtrlHDelete 

Ctrl+Backspace 

Shift+Tab 

Ctrl+U 

Ctrl+Shift+U 

Ctrl+Shift+End 选择 至 文档 末尾 
Ctrl+Shift+Home 选择 至 文档 末尾 开始 
Shift+End 选择 至 行 尾 
Shift+Home 选择 至 行 开 始 处 
Shift+tAlt+End 垂直 选择 到 最 后 尾 
Shift+Alt+Home 垂直 选择 到 最 前 面 
CtrlHA 全 选 

CtrlHW 选择 当前 单词 
Ctrl+ShifttPage Up 选择 至 本 页 前 面 
Ctrl+Shift+Page Down 选择 至 本 页 后 面 
Ctrl+End 文档 定位 到 最 后 
CtrlHHome 文档 定位 到 最 前 
Ctrl+G 转 到 ... 
CtrlHK，CtrlHP 上 一 个 标签 
CtrlHK，CtaHN 下 一 个 标签 


AltrF10 


调试 一 ApplyCodeChanges 


附录 Visual Studio 操作 快捷 键 
( 续 表 ) 
快 捷 键 功 能 
Ctrl+Alt+Break 停止 调试 
Ctrl+Shift+F9 取消 所 有 断 点 
CtrlHF9 允许 中 断 
Ctrl+F5 运行 不 调试 


下 面 分 类 介绍 一 些 快捷 键 ， 某 些 可 能 与 上 面 有 重复 ， 主 要 是 增加 读者 印象 。 


(1) 调试 快捷 键 列 表 。 


快 捷 键 功 能 
F6 生成 解决 方案 
Ctrl+F6 生成 当前 项 目 
F7 查看 代码 
Shift+F7 查看 窗 体 设计 器 
F5 启动 调试 
Ctrl+F5 开始 执行 〈 不 调试 ) 
Shift+F5 停止 调试 
Ctrl+Shift+F5 重启 调试 
F9 切换 断 点 
Ctrl+F9 启用 /停止 断 点 
Ctrl+Shift+F9 删除 全 部 断 点 
F10 逐 过 程 
Ctrl+F10 运行 到 光标 处 
Fl1 逐 语 句 
(2) 编辑 快捷 键 列 表 。 
快 捷 键 功 能 
Shift+Alt+Enter 切换 全 屏 编辑 
Ctl+B, T 切换 书签 开关 
Ctl+B, N 移动 到 下 一 书签 
CtrlHB，P 移动 到 上 一 书签 
CtrlHB，C 清除 全 部 标签 
CtrlHI 渐进 式 搜索 
CtrlHShiftHI 反 向 渐进 式 搜 索 
CtrHtF 查找 
Ctl+Shift+F 在 文件 中 查找 
F3 查找 下 一 个 
Shift+F3 查找 上 一 个 
CtrlHHH 替换 
CtrlHShiftHH 在 文件 中 替换 
Alt+F12 查找 符号 〈 列 出 所 有 查找 结果 ) 
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( 续 表 ) 
快 捷 键 功 能 
Ctl+Shift+rV 剪贴 板 循环 
Ctrl+ 左 右 箭头 键 一 次 可 以 移动 一 个 单词 
Ctrl+ 上 下 箭头 键 滚动 代码 屏幕 ， 但 不 移动 光标 位 置 
Ctrl+Shift+L 删除 当前 行 
CtrlHM，M 隐藏 或 展开 当前 嵌 套 的 折 释 状态 
CtrlHM， 工 将 所 有 过 程 设 置 为 相同 的 隐藏 或 展开 状态 
Ctl+M, P 停止 大 纲 显示 
Ctl+E, S 查看 空白 
Ctl+E, W 自动 换行 
CtltG 转 到 指定 行 
Shift+Alt+ 箭 头 键 选择 矩形 文本 
Alt+ 鼠 标 左 按钮 选择 矩形 文本 
Ctrl+Shift+U 全 部 变 为 大 写 
CtrlHU 全 部 变 为 小 写 
(3) 代码 快捷 键 列表 。 
快 捷 键 功 能 
CtrlHJ 列 出 成 员 
Ctrl+Shift+ 空 格 键 参数 信息 
Ctrl+K，I 快速 信息 
Ctl+E, C 注释 选 定 内 容 
Ctrl+E，U 取消 选 定 注释 内 容 
CtrlHK，M 生成 方法 存根 
CtritK, X 插入 代码 段 
Ctrl+K，S 插入 外 侧 代码 
(4) 有 关 窗 口 的 快捷 键 列表 。 
快捷 键 功 能 
CtrlHEHW，W 浏览 器 窗口 
Ctrlt+W, S 解决 方案 管理 器 
CtrlHW，C 类 视图 
Ctrl+W, E 错误 列表 
CtrlHW，O 输出 视图 
CtrlHW，P 属性 窗口 
CtrlHW，T 任务 列表 
Ctli+W, X 工具 箱 
CtrlHW，B 书签 窗口 
CtrlHW，TU 文档 大 纲 
Ctl+D, B 断 点 窗口 
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( 续 表 ) 
快 捷 键 功 能 
Ctl+D, I 即时 窗口 
Ctrit+Tab 活动 窗 体 切 换 
Ctrl+Shift+N 新 建 项 目 
Ctrl+Shift+tO 打开 项 目 
Ctrl+Shift+S 全 部 保存 
Shift+Alt+C 新 建 类 
CtrlH+ShiftrA 新 建 项 
(5) Visual Studio 中 没有 明确 指出 的 快捷 键 。 
快 捷 键 功 能 
直接 完成 类 或 函数 〈 本 来 这 个 并 不 算 隐 藏 的 快捷 
Ctrl+Space 键 , 但 是 因为 中 文 输入 法 抢占 了 这 个 快捷 键 , 替代 
的 快捷 键 是 Alt+Right) 
。 整 行 删除 , 并 且 将 这 一 行 放 到 剪贴 板 ( 这 时候 不 能 
Sh pelte 选中 一 段 内 容 ) 
. 粘贴 ， 使 用 CtrltV 的 功能 类 似 ， 主 要 是 为 了 和 
Shift+Insert 


CtrlHUp，Ctrl+Down 
Ctrl+BackSpace，Ctrl+Delete 


Ctrlt+Left, Ctrl+Right 


Shift+Delete 对 应 

滚动 编辑 器 ,但 尽量 不 移动 光标 ， 光 标 保证 在 可 见 
范围 内 

整 词 删除 ， 有 的 时 候 很 有 用 

按 整 词 移动 光标 (不 算 隐 藏 ， 和 前 面 儿 条 加 起 来 就 
是 Ctrl 光标 控制 套件 了 )》 

打开 执行 改名 , 实现 接口 和 抽象 类 的 小 窗口 (还 可 


eh 以 用 Ctrlt.， 不 过 有 的 中 文 输入 法 用 到 这 个 ) 
. 调试 是 打开 QuickWatch， 内 容 是 当前 光标 所 在 处 
Shift+F9 入 
的 内 容 
Shift+F12 查找 所 有 引用 
CtrlHF10=F5 开始 Debug 
Ctrl+F6 循环 查看 代码 窗口 ， 有 点 CtrlHTab 的 感觉 
Ctrl+F3 查找 当前 光标 选中 的 内 容 ， 可 以 和 F3 配合 使 用 
CtrlHF2 将 焦点 转移 到 类 的 下 拉 框 上 
Alt+F7 CtrlHTab， 下 一 个 文档 窗口 
Alt+F11 新 开 Visual Studio 并 编辑 宏 
Alt+F12 查找 ， 等 同 于 CILHF 


性 和 


